Skip to content

Commit 4b7130c

Browse files
fix(clerk-js): Cache active sessions with useFetch (clerk#2733)
* chore(repo): Upgrade next version on playground * fix(clerk-js): Don't refetch sessions on every page visit * chore(clerk-js): Handler arrays in `useFetch` * chore(clerk-js): Clear cache between tests --------- Co-authored-by: panteliselef <panteliselef@outlook.com>
1 parent 0d565e1 commit 4b7130c

File tree

5 files changed

+43
-40
lines changed

5 files changed

+43
-40
lines changed

.changeset/ten-dryers-share.md

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
---
2+
---

packages/clerk-js/src/ui/components/UserProfile/ActiveDevicesSection.tsx

+30-37
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,9 @@
11
import { useSession, useUser } from '@clerk/shared/react';
22
import type { SessionWithActivitiesResource } from '@clerk/types';
3-
import React from 'react';
43

54
import { Badge, Col, descriptors, Flex, Icon, localizationKeys, Text, useLocalizations } from '../../customizables';
65
import { FullHeightLoader, ProfileSection, ThreeDotsMenu } from '../../elements';
7-
import { Action } from '../../elements/Action';
8-
import { useLoadingStatus } from '../../hooks';
6+
import { useFetch, useLoadingStatus } from '../../hooks';
97
import { DeviceLaptop, DeviceMobile } from '../../icons';
108
import { mqu, type PropsOfComponent } from '../../styledSystem';
119
import { getRelativeToNowDateKey } from '../../utils';
@@ -14,11 +12,8 @@ import { currentSessionFirst } from './utils';
1412
export const ActiveDevicesSection = () => {
1513
const { user } = useUser();
1614
const { session } = useSession();
17-
const [sessionsWithActivities, setSessionsWithActivities] = React.useState<SessionWithActivitiesResource[]>([]);
1815

19-
React.useEffect(() => {
20-
void user?.getSessions().then(sa => setSessionsWithActivities(sa));
21-
}, [user]);
16+
const { data: sessions, isLoading } = useFetch(user?.getSessions, 'user-sessions');
2217

2318
return (
2419
<ProfileSection.Root
@@ -27,20 +22,22 @@ export const ActiveDevicesSection = () => {
2722
id='activeDevices'
2823
>
2924
<ProfileSection.ItemList id='activeDevices'>
30-
{!sessionsWithActivities.length && <FullHeightLoader />}
31-
{!!sessionsWithActivities.length &&
32-
sessionsWithActivities.sort(currentSessionFirst(session!.id)).map(sa => (
33-
<DeviceAccordion
25+
{isLoading ? (
26+
<FullHeightLoader />
27+
) : (
28+
sessions?.sort(currentSessionFirst(session!.id)).map(sa => (
29+
<DeviceItem
3430
key={sa.id}
3531
session={sa}
3632
/>
37-
))}
33+
))
34+
)}
3835
</ProfileSection.ItemList>
3936
</ProfileSection.Root>
4037
);
4138
};
4239

43-
const DeviceAccordion = ({ session }: { session: SessionWithActivitiesResource }) => {
40+
const DeviceItem = ({ session }: { session: SessionWithActivitiesResource }) => {
4441
const isCurrent = useSession().session?.id === session.id;
4542
const status = useLoadingStatus();
4643
const revoke = async () => {
@@ -52,30 +49,26 @@ const DeviceAccordion = ({ session }: { session: SessionWithActivitiesResource }
5249
};
5350

5451
return (
55-
<Action.Root>
56-
<Action.Closed value=''>
57-
<ProfileSection.Item
58-
id='activeDevices'
59-
elementDescriptor={descriptors.activeDeviceListItem}
60-
elementId={isCurrent ? descriptors.activeDeviceListItem.setId('current') : undefined}
61-
sx={t => ({
62-
alignItems: 'flex-start',
63-
padding: `${t.space.$2} ${t.space.$4}`,
64-
marginLeft: `-${t.space.$4}`,
65-
borderRadius: t.radii.$md,
66-
':hover': { backgroundColor: t.colors.$blackAlpha50 },
67-
})}
68-
>
69-
{status.isLoading && <FullHeightLoader />}
70-
{!status.isLoading && (
71-
<>
72-
<DeviceInfo session={session} />
73-
{!isCurrent && <ActiveDeviceMenu revoke={revoke} />}
74-
</>
75-
)}
76-
</ProfileSection.Item>
77-
</Action.Closed>
78-
</Action.Root>
52+
<ProfileSection.Item
53+
id='activeDevices'
54+
elementDescriptor={descriptors.activeDeviceListItem}
55+
elementId={isCurrent ? descriptors.activeDeviceListItem.setId('current') : undefined}
56+
sx={t => ({
57+
alignItems: 'flex-start',
58+
padding: `${t.space.$2} ${t.space.$4}`,
59+
marginLeft: `-${t.space.$4}`,
60+
borderRadius: t.radii.$md,
61+
':hover': { backgroundColor: t.colors.$blackAlpha50 },
62+
})}
63+
>
64+
{status.isLoading && <FullHeightLoader />}
65+
{!status.isLoading && (
66+
<>
67+
<DeviceInfo session={session} />
68+
{!isCurrent && <ActiveDeviceMenu revoke={revoke} />}
69+
</>
70+
)}
71+
</ProfileSection.Item>
7972
);
8073
};
8174

packages/clerk-js/src/ui/components/UserProfile/__tests__/SecurityPage.test.tsx

+8
Original file line numberDiff line numberDiff line change
@@ -3,12 +3,20 @@ import { describe, it } from '@jest/globals';
33
import { within } from '@testing-library/dom';
44

55
import { render, screen, waitFor } from '../../../../testUtils';
6+
import { clearFetchCache } from '../../../hooks';
67
import { bindCreateFixtures } from '../../../utils/test/createFixtures';
78
import { SecurityPage } from '../SecurityPage';
89

910
const { createFixtures } = bindCreateFixtures('UserProfile');
1011

1112
describe('SecurityPage', () => {
13+
/**
14+
* `<SecurityPage/>` internally uses useFetch which caches the results, be sure to clear the cache before each test
15+
*/
16+
beforeEach(() => {
17+
clearFetchCache();
18+
});
19+
1220
it('renders the component', async () => {
1321
const { wrapper, fixtures } = await createFixtures(f => {
1422
f.withUser({ email_addresses: ['test@clerk.com'] });

packages/clerk-js/src/ui/hooks/useFetch.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,7 @@ export const clearFetchCache = () => {
3131
requestCache = new Map<string, State>();
3232
};
3333

34-
const serialize = (obj: unknown) => JSON.stringify(obj);
34+
const serialize = (key: unknown) => (typeof key === 'string' ? key : JSON.stringify(key));
3535

3636
const useCache = <K = any, V = any>(
3737
key: K,
@@ -94,7 +94,7 @@ export const useFetch = <K, T>(
9494
fetcherRef.current!(params)
9595
.then(result => {
9696
if (typeof result !== 'undefined') {
97-
const data = typeof result === 'object' ? { ...result } : result;
97+
const data = Array.isArray(result) ? result : typeof result === 'object' ? { ...result } : result;
9898
setCache({
9999
data,
100100
isLoading: false,

playground/nextjs/package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@
1818
"@clerk/shared": "file:.yalc/@clerk/shared",
1919
"@clerk/themes": "file:.yalc/@clerk/themes",
2020
"@clerk/types": "file:.yalc/@clerk/types",
21-
"next": "^13.5.6",
21+
"next": "^14.1.0",
2222
"react": "^18.2.0",
2323
"react-dom": "^18.2.0"
2424
},

0 commit comments

Comments
 (0)