Skip to content

fix(clerk-js): Navigate to tasks when switching sessions #6273

New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Merged
merged 13 commits into from
Jul 11, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
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/tall-dolls-wish.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
'@clerk/clerk-js': patch
---

Navigate to tasks when switching sessions
4 changes: 2 additions & 2 deletions integration/templates/next-app-router/src/app/page.tsx
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { SignedIn, SignedOut, SignIn, UserButton, Protect } from '@clerk/nextjs';
import { Protect, SignedIn, SignedOut, SignIn, UserButton } from '@clerk/nextjs';
import Link from 'next/link';
import { ClientId } from './client-id';

Expand All @@ -11,7 +11,7 @@ export default function Home() {
<SignedOut>SignedOut</SignedOut>
<Protect fallback={'SignedOut from protect'}>SignedIn from protect</Protect>
<SignIn
path={'/'}
routing='hash'
signUpUrl={'/sign-up'}
/>
<ul>
Expand Down
96 changes: 96 additions & 0 deletions integration/tests/session-tasks-multi-session.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,96 @@
import { test } from '@playwright/test';

import { appConfigs } from '../presets';
import type { FakeUser } from '../testUtils';
import { createTestUtils, testAgainstRunningApps } from '../testUtils';

testAgainstRunningApps({ withEnv: [appConfigs.envs.withSessionTasks] })(
'session tasks multi-session flow @nextjs',
({ app }) => {
test.describe.configure({ mode: 'serial' });

let user1: FakeUser;
let user2: FakeUser;

test.beforeAll(async () => {
const u = createTestUtils({ app });

user1 = u.services.users.createFakeUser();
user2 = u.services.users.createFakeUser();

await u.services.users.createBapiUser(user1);
await u.services.users.createBapiUser(user2);
});

test.afterAll(async () => {
const u = createTestUtils({ app });
await user1.deleteIfExists();
await user2.deleteIfExists();
await u.services.organizations.deleteAll();
await app.teardown();
});

test.afterEach(async ({ page, context }) => {
const u = createTestUtils({ app, page, context });
await u.page.signOut();
await u.page.context().clearCookies();
});

test('when switching sessions, navigate to task', async ({ page, context }) => {
const u = createTestUtils({ app, page, context });

// Performs sign-in
await u.po.signIn.goTo();
await u.po.signIn.setIdentifier(user1.email);
await u.po.signIn.continue();
await u.po.signIn.setPassword(user1.password);
await u.po.signIn.continue();
await u.po.expect.toBeSignedIn();

// Resolves task
const fakeOrganization = Object.assign(u.services.organizations.createFakeOrganization(), {
slug: u.services.organizations.createFakeOrganization().slug + '-with-session-tasks',
});

await u.po.sessionTask.resolveForceOrganizationSelectionTask(fakeOrganization);
await u.po.expect.toHaveResolvedTask();

// Navigates to after sign-in
await u.page.waitForAppUrl('/');

// Create second user, to initiate a pending session
// Don't resolve task and switch to active session afterwards
await u.po.signIn.goTo();
await u.po.signIn.setIdentifier(user2.email);
await u.po.signIn.continue();
await u.po.signIn.setPassword(user2.password);
await u.po.signIn.continue();

// Sign-in again back with active session
await u.po.signIn.goTo();
await u.po.signIn.setIdentifier(user1.email);
await u.po.signIn.continue();
await u.po.signIn.setPassword(user1.password);
await u.po.signIn.continue();

// Navigate to protected page, with active session, where user button gets rendered
await u.page.goToRelative('/user-button');

// Switch account, to a session that has a pending status
await u.po.userButton.waitForMounted();
await u.po.userButton.toggleTrigger();
await u.po.userButton.waitForPopover();
await u.po.userButton.toHaveVisibleMenuItems([/Manage account/i, /Sign out$/i]);
await u.po.userButton.switchAccount(user2.email);

// Resolve task
await u.po.signIn.waitForMounted();
const fakeOrganization2 = u.services.organizations.createFakeOrganization();
await u.po.sessionTask.resolveForceOrganizationSelectionTask(fakeOrganization2);
await u.po.expect.toHaveResolvedTask();

// Navigates to after sign-in
await u.page.waitForAppUrl('/');
});
},
);
45 changes: 28 additions & 17 deletions integration/tests/session-tasks-sign-in.test.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { createClerkClient } from '@clerk/backend';
import { expect, test } from '@playwright/test';
import { test } from '@playwright/test';

import { appConfigs } from '../presets';
import { instanceKeys } from '../presets/envs';
Expand All @@ -12,46 +12,54 @@ testAgainstRunningApps({ withEnv: [appConfigs.envs.withSessionTasks] })(
({ app }) => {
test.describe.configure({ mode: 'serial' });

let fakeUser: FakeUser;
let user: FakeUser;

test.beforeAll(async () => {
const u = createTestUtils({ app });
fakeUser = u.services.users.createFakeUser();
await u.services.users.createBapiUser(fakeUser);
user = u.services.users.createFakeUser();
await u.services.users.createBapiUser(user);
});

test.afterAll(async () => {
const u = createTestUtils({ app });
await fakeUser.deleteIfExists();
await user.deleteIfExists();
await u.services.organizations.deleteAll();
await app.teardown();
});

test.skip('with email and password, navigate to task on after sign-in', async ({ page, context }) => {
test.afterEach(async ({ page, context }) => {
const u = createTestUtils({ app, page, context });
await u.page.signOut();
await u.page.context().clearCookies();
});

test('with email and password, navigate to task on after sign-in', async ({ page, context }) => {
const u = createTestUtils({ app, page, context });

// Performs sign-in
await u.po.signIn.goTo();
await u.po.signIn.setIdentifier(fakeUser.email);
await u.po.signIn.setIdentifier(user.email);
await u.po.signIn.continue();
await u.po.signIn.setPassword(fakeUser.password);
await u.po.signIn.setPassword(user.password);
await u.po.signIn.continue();
await u.po.expect.toBeSignedIn();

// Redirects back to tasks when accessing protected route by `auth.protect`
await u.page.goToRelative('/page-protected');
expect(page.url()).toContain('tasks');

// Resolves task
const fakeOrganization = u.services.organizations.createFakeOrganization();
const fakeOrganization = Object.assign(u.services.organizations.createFakeOrganization(), {
slug: u.services.organizations.createFakeOrganization().slug + '-with-sign-in-password',
});
await u.po.signIn.waitForMounted();
await u.po.sessionTask.resolveForceOrganizationSelectionTask(fakeOrganization);
await u.po.expect.toHaveResolvedTask();

// Navigates to after sign-in
await u.page.waitForAppUrl('/');
});

test.skip('with sso, navigate to task on after sign-in', async ({ page, context }) => {
test('with sso, navigate to task on after sign-in', async ({ page, context }) => {
const u = createTestUtils({ app, page, context });

// Create a clerkClient for the OAuth provider instance
Expand All @@ -60,32 +68,35 @@ testAgainstRunningApps({ withEnv: [appConfigs.envs.withSessionTasks] })(
publishableKey: instanceKeys.get('oauth-provider').pk,
});
const users = createUserService(client);
fakeUser = users.createFakeUser({
const userFromOAuth = users.createFakeUser({
withUsername: true,
});
// Create the user on the OAuth provider instance so we do not need to sign up twice
await users.createBapiUser(fakeUser);
await users.createBapiUser(userFromOAuth);

// Performs sign-in with SSO
await u.po.signIn.goTo();
await u.page.getByRole('button', { name: 'E2E OAuth Provider' }).click();
await u.page.getByText('Sign in to oauth-provider').waitFor();
await u.po.signIn.setIdentifier(fakeUser.email);
await u.po.signIn.setIdentifier(userFromOAuth.email);
await u.po.signIn.continue();
await u.po.signIn.enterTestOtpCode();

// Resolves task
const fakeOrganization = u.services.organizations.createFakeOrganization();
await u.po.signIn.waitForMounted();
const fakeOrganization = Object.assign(u.services.organizations.createFakeOrganization(), {
slug: u.services.organizations.createFakeOrganization().slug + '-with-sign-in-sso',
});
await u.po.sessionTask.resolveForceOrganizationSelectionTask(fakeOrganization);
await u.po.expect.toHaveResolvedTask();

// Navigates to after sign-in
await u.page.waitForAppUrl('/');

// Delete the user on the OAuth provider instance
await fakeUser.deleteIfExists();
await userFromOAuth.deleteIfExists();
// Delete the user on the app instance.
await u.services.users.deleteIfExists({ email: fakeUser.email });
await u.services.users.deleteIfExists({ email: userFromOAuth.email });
});
},
);
46 changes: 4 additions & 42 deletions integration/tests/session-tasks-sign-up.test.ts
Original file line number Diff line number Diff line change
@@ -1,11 +1,8 @@
import { expect, test } from '@playwright/test';

import { createClerkClient } from '@clerk/backend';
import { appConfigs } from '../presets';
import { instanceKeys } from '../presets/envs';
import type { FakeUser } from '../testUtils';
import { createTestUtils, testAgainstRunningApps } from '../testUtils';
import { createUserService } from '../testUtils/usersService';

testAgainstRunningApps({ withEnv: [appConfigs.envs.withSessionTasks] })(
'session tasks after sign-up flow @nextjs',
Expand All @@ -30,7 +27,7 @@ testAgainstRunningApps({ withEnv: [appConfigs.envs.withSessionTasks] })(
await app.teardown();
});

test.skip('navigate to task on after sign-up', async ({ page, context }) => {
test('navigate to task on after sign-up', async ({ page, context }) => {
// Performs sign-up
const u = createTestUtils({ app, page, context });
await u.po.signUp.goTo();
Expand All @@ -42,52 +39,17 @@ testAgainstRunningApps({ withEnv: [appConfigs.envs.withSessionTasks] })(

// Redirects back to tasks when accessing protected route by `auth.protect`
await u.page.goToRelative('/page-protected');
expect(page.url()).toContain('tasks');
expect(u.page.url()).toContain('tasks');

// Resolves task
const fakeOrganization = u.services.organizations.createFakeOrganization();
await u.po.sessionTask.resolveForceOrganizationSelectionTask(fakeOrganization);
await u.po.expect.toHaveResolvedTask();

// Navigates to after sign-up
await u.page.waitForAppUrl('/');
});

test.skip('with sso, navigate to task on after sign-up', async ({ page, context }) => {
const u = createTestUtils({ app, page, context });

// Create a clerkClient for the OAuth provider instance
const client = createClerkClient({
secretKey: instanceKeys.get('oauth-provider').sk,
publishableKey: instanceKeys.get('oauth-provider').pk,
});
const users = createUserService(client);
fakeUser = users.createFakeUser({
withUsername: true,
const fakeOrganization = Object.assign(u.services.organizations.createFakeOrganization(), {
slug: u.services.organizations.createFakeOrganization().slug + '-with-sign-up',
});
// Create the user on the OAuth provider instance so we do not need to sign up twice
await users.createBapiUser(fakeUser);

// Performs sign-up (transfer flow with sign-in) with SSO
await u.po.signIn.goTo();
await u.page.getByRole('button', { name: 'E2E OAuth Provider' }).click();
await u.page.getByText('Sign in to oauth-provider').waitFor();
await u.po.signIn.setIdentifier(fakeUser.email);
await u.po.signIn.continue();
await u.po.signIn.enterTestOtpCode();

// Resolves task
const fakeOrganization = u.services.organizations.createFakeOrganization();
await u.po.sessionTask.resolveForceOrganizationSelectionTask(fakeOrganization);
await u.po.expect.toHaveResolvedTask();

// Navigates to after sign-up
await u.page.waitForAppUrl('/');

// Delete the user on the OAuth provider instance
await fakeUser.deleteIfExists();
// Delete the user on the app instance.
await u.services.users.deleteIfExists({ email: fakeUser.email });
});
},
);
2 changes: 1 addition & 1 deletion packages/clerk-js/bundlewatch.config.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"files": [
{ "path": "./dist/clerk.js", "maxSize": "612.37kB" },
{ "path": "./dist/clerk.js", "maxSize": "614kB" },
{ "path": "./dist/clerk.browser.js", "maxSize": "72.2KB" },
{ "path": "./dist/clerk.legacy.browser.js", "maxSize": "115KB" },
{ "path": "./dist/clerk.headless*.js", "maxSize": "55KB" },
Expand Down
19 changes: 11 additions & 8 deletions packages/clerk-js/src/core/clerk.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1209,16 +1209,17 @@ export class Clerk implements ClerkInterface {
}
}

if (newSession?.status === 'pending') {
await this.#handlePendingSession(newSession);
return;
}

/**
* Hint to each framework, that the user will be signed out when `{session: null}` is provided.
*/
await onBeforeSetActive(newSession === null ? 'sign-out' : undefined);

if (newSession?.status === 'pending') {
await this.#handlePendingSession(newSession);
await onAfterSetActive();
Comment on lines +1218 to +1219
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We weren't revalidating the server state on setActive with a pending session, fixing it now.

return;
}

//1. setLastActiveSession to passed user session (add a param).
// Note that this will also update the session's active organization
// id.
Expand Down Expand Up @@ -1301,8 +1302,10 @@ export class Clerk implements ClerkInterface {
eventBus.emit(events.TokenUpdate, { token: null });
}

// Only triggers navigation for internal routing, in order to not affect custom flows
if (newSession?.currentTask && this.#componentNavigationContext) {
// Only triggers navigation for internal AIO components routing or multi-session switch
const isSwitchingSessions = this.session?.id != session.id;
const shouldNavigateOnSetActive = this.#componentNavigationContext || isSwitchingSessions;
if (newSession?.currentTask && shouldNavigateOnSetActive) {
await navigateToTask(session.currentTask.key, {
options: this.#options,
environment: this.environment,
Expand All @@ -1317,7 +1320,7 @@ export class Clerk implements ClerkInterface {

public __experimental_navigateToTask = async ({ redirectUrlComplete }: NextTaskParams = {}): Promise<void> => {
/**
* Invalidate previously cache pages with auth state before navigating
* Invalidate previously cached pages with auth state before navigating
*/
const onBeforeSetActive: SetActiveHook =
typeof window !== 'undefined' && typeof window.__unstable__onBeforeSetActive === 'function'
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,9 @@ export const createUserButtonPageObject = (testArgs: { page: EnhancedPage }) =>
triggerManageAccount: () => {
return page.getByRole('menuitem', { name: /Manage account/i }).click();
},
switchAccount: (emailAddress: string) => {
return page.getByText(emailAddress).click();
},
};

return self;
Expand Down
Loading