Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
40 commits
Select commit Hold shift + click to select a range
a28fa4d
feat(clerk-js): Lazy query client
panteliselef Sep 10, 2025
a3ad75c
wip
panteliselef Sep 19, 2025
e6d6561
update lock file
panteliselef Oct 3, 2025
82901ad
wip
panteliselef Oct 3, 2025
f5c27e3
feat(shared): `useSubscription` variant with React Query
panteliselef Oct 3, 2025
827f536
cleanup
panteliselef Oct 3, 2025
96b68d2
bundlewatch and changeset
panteliselef Oct 3, 2025
c07fe17
fix lint
panteliselef Oct 3, 2025
7fc1248
wip
panteliselef Oct 3, 2025
7f5cb97
Revert "wip"
panteliselef Oct 14, 2025
6da5615
handle init state
panteliselef Oct 14, 2025
992eda8
add support for infinite
panteliselef Oct 14, 2025
f9b7f64
wip some a few test continue to fail
panteliselef Oct 14, 2025
f8bee5d
mock queryClient
panteliselef Oct 14, 2025
eeae1c5
Merge branch 'refs/heads/main' into elef/replace-swr-with-rq-infinite
panteliselef Nov 4, 2025
73e9101
fix conflicts about `usePagesOrInfinite.swr`
panteliselef Nov 4, 2025
751dd31
temp changeset
panteliselef Nov 4, 2025
582f018
allow unit tests to run with rq variant
panteliselef Nov 4, 2025
47f6649
run `useSubscription` tests locally for RQ variant
panteliselef Nov 4, 2025
fad9ac7
wip
panteliselef Nov 5, 2025
83a1a06
Revert "wip"
panteliselef Nov 5, 2025
7883fb3
fix(shared): Avoid revalidating first page on infinite pagination
panteliselef Nov 5, 2025
5ee3992
wip
panteliselef Nov 5, 2025
8e39385
Revert "wip"
panteliselef Nov 5, 2025
9ff7e2a
address skipped tests
panteliselef Nov 5, 2025
9e7d21d
typos
panteliselef Nov 5, 2025
4a68a52
wip
panteliselef Nov 5, 2025
f7a21ac
fix usePlans
panteliselef Nov 5, 2025
b8ea10d
wip
panteliselef Nov 5, 2025
57cb706
wip
panteliselef Nov 5, 2025
631e5e5
wip
panteliselef Nov 5, 2025
a732df6
all pass
panteliselef Nov 5, 2025
2d3c550
fix swr not passing params in infinite mode
panteliselef Nov 5, 2025
1b4c7f0
improve flakiness
panteliselef Nov 5, 2025
8d90888
Merge branch 'main' into elef/replace-swr-with-rq-infinite
panteliselef Nov 5, 2025
6f27804
improve flakiness
panteliselef Nov 5, 2025
a9df03b
improve mocks
panteliselef Nov 5, 2025
35f4e83
update CI to include RQ variants
panteliselef Nov 5, 2025
0ffffa2
Merge branch 'main' into elef/replace-swr-with-rq-infinite
panteliselef Nov 10, 2025
031083c
cleanup
panteliselef Nov 10, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions .changeset/fuzzy-keys-smell.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
'@clerk/shared': patch
---

Build internal variants of all paginated hooks that use React Query instead of SWR.
64 changes: 53 additions & 11 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -191,7 +191,7 @@ jobs:
unit-tests:
needs: [check-permissions, build-packages]
if: ${{ github.event_name != 'pull_request' || github.event.pull_request.draft == false }}
name: Unit Tests
name: Unit Tests (${{ matrix.node-version }}, ${{ matrix.filter-label }}${{ matrix.clerk-use-rq == 'true' && ', RQ' || '' }})
permissions:
contents: read
actions: write # needed for actions/upload-artifact
Expand All @@ -205,11 +205,17 @@ jobs:
TURBO_SUMMARIZE: false

strategy:
fail-fast: true
fail-fast: false
matrix:
include:
- node-version: 22
test-filter: "**"
clerk-use-rq: "false"
filter-label: "**"
- node-version: 22
test-filter: "--filter=@clerk/shared --filter=@clerk/clerk-js"
clerk-use-rq: "true"
filter-label: "shared, clerk-js"

steps:
- name: Checkout Repo
Expand All @@ -229,22 +235,35 @@ jobs:
turbo-team: ${{ vars.TURBO_TEAM }}
turbo-token: ${{ secrets.TURBO_TOKEN }}

- name: Rebuild @clerk/shared with CLERK_USE_RQ=true
if: ${{ matrix.clerk-use-rq == 'true' }}
run: pnpm turbo build $TURBO_ARGS --filter=@clerk/shared --force
env:
CLERK_USE_RQ: true

- name: Rebuild dependent packages with CLERK_USE_RQ=true
if: ${{ matrix.clerk-use-rq == 'true' }}
run: pnpm turbo build $TURBO_ARGS --filter=@clerk/shared^... --force
env:
CLERK_USE_RQ: true

- name: Run tests in packages
run: |
if [ "${{ matrix.test-filter }}" = "**" ]; then
echo "Running full test suite on Node ${{ matrix.node-version }}."
echo "Running full test suite on Node ${{ matrix.node-version }}"
pnpm turbo test $TURBO_ARGS
else
echo "Running LTS subset on Node ${{ matrix.node-version }}."
echo "Running tests: ${{ matrix.filter-label }}"
pnpm turbo test $TURBO_ARGS ${{ matrix.test-filter }}
fi
env:
NODE_VERSION: ${{ matrix.node-version }}
CLERK_USE_RQ: ${{ matrix.clerk-use-rq }}

- name: Run Typedoc tests
run: |
# Only run Typedoc tests for one matrix version
if [ "${{ matrix.node-version }}" == "22" ]; then
# Only run Typedoc tests for one matrix version and main test run
if [ "${{ matrix.node-version }}" == "22" ] && [ "${{ matrix.test-filter }}" = "**" ]; then
pnpm test:typedoc
fi
env:
Expand All @@ -255,14 +274,14 @@ jobs:
if: ${{ env.TURBO_SUMMARIZE == 'true' }}
continue-on-error: true
with:
name: turbo-summary-report-unit-${{ github.run_id }}-${{ github.run_attempt }}-node-${{ matrix.node-version }}
name: turbo-summary-report-unit-${{ github.run_id }}-${{ github.run_attempt }}-node-${{ matrix.node-version }}${{ matrix.clerk-use-rq == 'true' && '-rq' || '' }}
path: .turbo/runs
retention-days: 5

integration-tests:
needs: [check-permissions, build-packages]
if: ${{ github.event_name != 'pull_request' || github.event.pull_request.draft == false }}
name: Integration Tests
name: Integration Tests (${{ matrix.test-name }}, ${{ matrix.test-project }}${{ matrix.next-version && format(', {0}', matrix.next-version) || '' }}${{ matrix.clerk-use-rq == 'true' && ', RQ' || '' }})
permissions:
contents: read
actions: write # needed for actions/upload-artifact
Expand Down Expand Up @@ -291,18 +310,28 @@ jobs:
'vue',
'nuxt',
'react-router',
'billing',
'machine',
'custom',
]
test-project: ["chrome"]
include:
- test-name: 'billing'
test-project: 'chrome'
clerk-use-rq: 'false'
- test-name: 'billing'
test-project: 'chrome'
clerk-use-rq: 'true'
- test-name: 'nextjs'
test-project: 'chrome'
next-version: '14'
- test-name: 'nextjs'
test-project: 'chrome'
next-version: '15'
clerk-use-rq: 'false'
- test-name: 'nextjs'
test-project: 'chrome'
next-version: '15'
clerk-use-rq: 'true'
- test-name: 'nextjs'
test-project: 'chrome'
next-version: '16'
Expand Down Expand Up @@ -360,12 +389,24 @@ jobs:
echo "affected=${AFFECTED}"
echo "affected=${AFFECTED}" >> $GITHUB_OUTPUT

- name: Rebuild @clerk/shared with CLERK_USE_RQ=true
if: ${{ steps.task-status.outputs.affected == '1' && matrix.clerk-use-rq == 'true' }}
run: pnpm turbo build $TURBO_ARGS --filter=@clerk/shared --force
env:
CLERK_USE_RQ: true

- name: Rebuild dependent packages with CLERK_USE_RQ=true
if: ${{ steps.task-status.outputs.affected == '1' && matrix.clerk-use-rq == 'true' }}
run: pnpm turbo build $TURBO_ARGS --filter=@clerk/shared^... --force
env:
CLERK_USE_RQ: true

- name: Verdaccio
if: ${{ steps.task-status.outputs.affected == '1' }}
uses: ./.github/actions/verdaccio
with:
publish-cmd: |
if [ "$(pnpm config get registry)" = "https://registry.npmjs.org/" ]; then echo 'Error: Using default registry' && exit 1; else pnpm turbo build $TURBO_ARGS --only && pnpm changeset publish --no-git-tag; fi
if [ "$(pnpm config get registry)" = "https://registry.npmjs.org/" ]; then echo 'Error: Using default registry' && exit 1; else CLERK_USE_RQ=${{ matrix.clerk-use-rq }} pnpm turbo build $TURBO_ARGS --only && pnpm changeset publish --no-git-tag; fi

- name: Edit .npmrc [link-workspace-packages=false]
run: sed -i -E 's/link-workspace-packages=(deep|true)/link-workspace-packages=false/' .npmrc
Expand Down Expand Up @@ -425,6 +466,7 @@ jobs:
E2E_NEXTJS_VERSION: ${{ matrix.next-version }}
E2E_PROJECT: ${{ matrix.test-project }}
E2E_CLERK_ENCRYPTION_KEY: ${{ matrix.clerk-encryption-key }}
CLERK_USE_RQ: ${{ matrix.clerk-use-rq }}
INTEGRATION_INSTANCE_KEYS: ${{ secrets.INTEGRATION_INSTANCE_KEYS }}
MAILSAC_API_KEY: ${{ secrets.MAILSAC_API_KEY }}
NODE_EXTRA_CA_CERTS: ${{ github.workspace }}/integration/certs/rootCA.pem
Expand All @@ -433,7 +475,7 @@ jobs:
if: ${{ cancelled() || failure() }}
uses: actions/upload-artifact@v4
with:
name: playwright-traces-${{ github.run_id }}-${{ github.run_attempt }}-${{ matrix.test-name }}
name: playwright-traces-${{ github.run_id }}-${{ github.run_attempt }}-${{ matrix.test-name }}${{ matrix.next-version && format('-next{0}', matrix.next-version) || '' }}${{ matrix.clerk-use-rq == 'true' && '-rq' || '' }}
path: integration/test-results
retention-days: 1

Expand Down
22 changes: 22 additions & 0 deletions packages/clerk-js/src/test/mock-helpers.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import type { ActiveSessionResource, LoadedClerk } from '@clerk/shared/types';
import { type Mocked, vi } from 'vitest';

import { QueryClient } from '../core/query-core';
import type { RouteContextValue } from '../ui/router';

type FunctionLike = (...args: any) => any;
Expand Down Expand Up @@ -45,6 +46,20 @@ export const mockClerkMethods = (clerk: LoadedClerk): DeepVitestMocked<LoadedCle
// Cast clerk to any to allow mocking properties
const clerkAny = clerk as any;

const defaultQueryClient = {
__tag: 'clerk-rq-client' as const,
client: new QueryClient({
defaultOptions: {
queries: {
retry: false,
// Setting staleTime to Infinity will not cause issues between tests as long as each test
// case has its own wrapper that initializes a Clerk instance with a new QueryClient.
staleTime: Infinity,
},
},
}),
};

mockMethodsOf(clerkAny);
if (clerkAny.client) {
mockMethodsOf(clerkAny.client.signIn);
Expand Down Expand Up @@ -76,6 +91,13 @@ export const mockClerkMethods = (clerk: LoadedClerk): DeepVitestMocked<LoadedCle
if (clerkAny.billing) {
mockMethodsOf(clerkAny.billing);
}

// Mock the __internal_queryClient getter property
Object.defineProperty(clerkAny, '__internal_queryClient', {
get: vi.fn(() => defaultQueryClient),
configurable: true,
});

mockProp(clerkAny, 'navigate');
mockProp(clerkAny, 'setActive');
mockProp(clerkAny, 'redirectWithAuth');
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -316,6 +316,12 @@ describe('Checkout', () => {
});

const freeTrialEndsAt = new Date('2025-08-19');

fixtures.clerk.user?.getPaymentMethods.mockResolvedValue({
data: [],
total_count: 0,
});

fixtures.clerk.billing.startCheckout.mockResolvedValue({
id: 'chk_trial_1',
status: 'needs_confirmation',
Expand Down Expand Up @@ -1033,13 +1039,18 @@ describe('Checkout', () => {
{ wrapper },
);

await waitFor(async () => {
await waitFor(() => {
expect(getByRole('heading', { name: 'Checkout' })).toBeVisible();
const addPaymentMethodButton = getByText('Add payment method');
expect(addPaymentMethodButton).toBeVisible();
await userEvent.click(addPaymentMethodButton);
});

const addPaymentMethodButton = await waitFor(() => {
const button = getByText('Add payment method');
expect(button).toBeVisible();
return button;
});

await userEvent.click(addPaymentMethodButton);

await waitFor(() => {
expect(getByRole('button', { name: 'Start free trial' })).toBeInTheDocument();
});
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -45,7 +45,7 @@ describe('OrganizationList', () => {
});
});

fixtures.clerk.user?.getOrganizationMemberships.mockReturnValueOnce(
fixtures.clerk.user?.getOrganizationMemberships.mockReturnValue(
Promise.resolve({
data: [
createFakeUserOrganizationMembership({
Expand Down Expand Up @@ -117,7 +117,7 @@ describe('OrganizationList', () => {
});
});

fixtures.clerk.user?.getOrganizationMemberships.mockReturnValueOnce(
fixtures.clerk.user?.getOrganizationMemberships.mockReturnValue(
Promise.resolve({
data: [
createFakeUserOrganizationMembership({
Expand Down Expand Up @@ -156,7 +156,7 @@ describe('OrganizationList', () => {
}),
);

fixtures.clerk.user?.getOrganizationInvitations.mockReturnValueOnce(
fixtures.clerk.user?.getOrganizationInvitations.mockReturnValue(
Promise.resolve({
data: [invitation],
total_count: 1,
Expand Down Expand Up @@ -342,7 +342,7 @@ describe('OrganizationList', () => {
});

await waitFor(async () => {
fixtures.clerk.setActive.mockReturnValueOnce(Promise.resolve());
fixtures.clerk.setActive.mockReturnValue(Promise.resolve());
await userEvent.click(getByText(/Personal account/i));

expect(fixtures.router.navigate).toHaveBeenCalledWith(`/user/test_user_id`);
Expand Down Expand Up @@ -376,7 +376,7 @@ describe('OrganizationList', () => {
},
});

fixtures.clerk.user?.getOrganizationMemberships.mockReturnValueOnce(
fixtures.clerk.user?.getOrganizationMemberships.mockReturnValue(
Promise.resolve({
data: [membership],
total_count: 1,
Expand All @@ -392,7 +392,7 @@ describe('OrganizationList', () => {
});

await waitFor(async () => {
fixtures.clerk.setActive.mockReturnValueOnce(Promise.resolve());
fixtures.clerk.setActive.mockReturnValue(Promise.resolve());
await userEvent.click(getByRole('button', { name: /Org1/i }));
expect(fixtures.clerk.setActive).toHaveBeenCalledWith(
expect.objectContaining({
Expand Down Expand Up @@ -423,7 +423,7 @@ describe('OrganizationList', () => {
wrapper,
});

fixtures.clerk.setActive.mockReturnValueOnce(Promise.resolve());
fixtures.clerk.setActive.mockReturnValue(Promise.resolve());
await waitFor(async () =>
expect(await findByRole('menuitem', { name: 'Create organization' })).toBeInTheDocument(),
);
Expand Down
Loading
Loading