From 15bade2be215dac94a34069bdf8309ca7287fb37 Mon Sep 17 00:00:00 2001 From: Mihaly Lengyel Date: Wed, 19 Jun 2024 13:33:20 +0200 Subject: [PATCH] feat: skip unnecessary updates in SessionAuth --- CHANGELOG.md | 6 ++++ examples/for-tests-react-16/src/App.js | 8 +++++ .../for-tests-react-16/src/testContext.js | 1 + examples/for-tests/src/App.js | 8 +++++ examples/for-tests/src/testContext.js | 1 + lib/build/index2.js | 34 ++++++++++++++----- lib/ts/recipe/session/sessionAuth.tsx | 34 ++++++++++++++----- test/end-to-end/signin.test.js | 29 ++++++++++++++-- 8 files changed, 101 insertions(+), 20 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 19d940483..69b40c159 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -7,6 +7,12 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [unreleased] +## [0.42.3] - 2024-07-19 + +### Changes + +- Now we only update the session context if the object changes by value. This optimization should help reduce unnecessary re-renders. + ## [0.42.2] - 2024-05-29 ### Changes diff --git a/examples/for-tests-react-16/src/App.js b/examples/for-tests-react-16/src/App.js index 41c323070..e5b4cf428 100644 --- a/examples/for-tests-react-16/src/App.js +++ b/examples/for-tests-react-16/src/App.js @@ -418,6 +418,14 @@ export const DashboardNoAuthRequired = doNotUseReactRouterDom export function DashboardNoAuthRequiredHelper(props) { let sessionContext = useSessionContext(); + useEffect(() => { + if (testContext.signoutOnSessionNotExists) { + if (!sessionContext.loading && !sessionContext.doesSessionExist) { + Session.signOut(); + } + } + }, [sessionContext]); + if (sessionContext.loading) { return null; } diff --git a/examples/for-tests-react-16/src/testContext.js b/examples/for-tests-react-16/src/testContext.js index 0d1a4caf3..a580fbd13 100644 --- a/examples/for-tests-react-16/src/testContext.js +++ b/examples/for-tests-react-16/src/testContext.js @@ -21,6 +21,7 @@ export function getTestContext() { : undefined, enableMFA: localStorage.getItem("enableMFA") === "true", defaultToEmail: localStorage.getItem("defaultToEmail") !== "false", + signoutOnSessionNotExists: localStorage.getItem("signoutOnSessionNotExists") === "true", disableRedirectionAfterSuccessfulSignInUp: localStorage.getItem("disableRedirectionAfterSuccessfulSignInUp") === "true", }; diff --git a/examples/for-tests/src/App.js b/examples/for-tests/src/App.js index 048337ec8..330ab59b2 100644 --- a/examples/for-tests/src/App.js +++ b/examples/for-tests/src/App.js @@ -590,6 +590,14 @@ export const DashboardNoAuthRequired = doNotUseReactRouterDom export function DashboardNoAuthRequiredHelper(props) { let sessionContext = useSessionContext(); + useEffect(() => { + if (testContext.signoutOnSessionNotExists) { + if (!sessionContext.loading && !sessionContext.doesSessionExist) { + Session.signOut(); + } + } + }, [sessionContext]); + if (sessionContext.loading) { return null; } diff --git a/examples/for-tests/src/testContext.js b/examples/for-tests/src/testContext.js index 02a14c3a1..0651c6aa8 100644 --- a/examples/for-tests/src/testContext.js +++ b/examples/for-tests/src/testContext.js @@ -22,6 +22,7 @@ export function getTestContext() { }, enableMFA: localStorage.getItem("enableMFA") === "true", defaultToEmail: localStorage.getItem("defaultToEmail") !== "false", + signoutOnSessionNotExists: localStorage.getItem("signoutOnSessionNotExists") === "true", disableRedirectionAfterSuccessfulSignInUp: localStorage.getItem("disableRedirectionAfterSuccessfulSignInUp") === "true", }; diff --git a/lib/build/index2.js b/lib/build/index2.js index e5fb3ff39..999e98241 100644 --- a/lib/build/index2.js +++ b/lib/build/index2.js @@ -2057,6 +2057,22 @@ var SessionAuth = function (_a) { var _c = React.useState({ loading: true }), context = _c[0], setContext = _c[1]; + var setContextIfChanged = React.useCallback( + function (newValue) { + setContext(function (oldValue) { + // We can't do this check before re-validation because there are be validators that depend on the current time + // Since the context is constructed by the same functions the property order should be stable, meaning that + // a simple JSON string check should be sufficient. + // Plus since this is just an optimization it is fine to have false positives, + // and this method won't have false negatives (where we'd miss an update). + if (JSON.stringify(oldValue) !== JSON.stringify(newValue)) { + return newValue; + } + return oldValue; + }); + }, + [setContext] + ); var session = React.useRef(); // We store this here, to prevent the list of called hooks changing even if a navigate hook is added later to SuperTokens. var navigateHookRef = React.useRef( @@ -2239,7 +2255,7 @@ var SessionAuth = function (_a) { ) ) return [3 /*break*/, 3]; - setContext(toSetContext); + setContextIfChanged(toSetContext); return [2 /*return*/]; case 3: return [ @@ -2268,7 +2284,7 @@ var SessionAuth = function (_a) { }); return [ 2 /*return*/, - setContext( + setContextIfChanged( genericComponentOverrideContext.__assign( genericComponentOverrideContext.__assign({}, toSetContext), { accessDeniedValidatorError: failureRedirectInfo.failedClaim } @@ -2278,7 +2294,7 @@ var SessionAuth = function (_a) { } _a.label = 8; case 8: - setContext(toSetContext); + setContextIfChanged(toSetContext); return [2 /*return*/]; } }); @@ -2353,7 +2369,7 @@ var SessionAuth = function (_a) { ) ) return [3 /*break*/, 5]; - setContext( + setContextIfChanged( genericComponentOverrideContext.__assign( genericComponentOverrideContext.__assign({}, event.sessionContext), { loading: false, invalidClaims: invalidClaims } @@ -2387,7 +2403,7 @@ var SessionAuth = function (_a) { }); return [ 2 /*return*/, - setContext( + setContextIfChanged( genericComponentOverrideContext.__assign( genericComponentOverrideContext.__assign({}, event.sessionContext), { @@ -2401,7 +2417,7 @@ var SessionAuth = function (_a) { } _b.label = 10; case 10: - setContext( + setContextIfChanged( genericComponentOverrideContext.__assign( genericComponentOverrideContext.__assign({}, event.sessionContext), { loading: false, invalidClaims: invalidClaims } @@ -2409,7 +2425,7 @@ var SessionAuth = function (_a) { ); return [2 /*return*/]; case 11: - setContext( + setContextIfChanged( genericComponentOverrideContext.__assign( genericComponentOverrideContext.__assign({}, event.sessionContext), { loading: false, invalidClaims: [] } @@ -2417,7 +2433,7 @@ var SessionAuth = function (_a) { ); return [2 /*return*/]; case 12: - setContext( + setContextIfChanged( genericComponentOverrideContext.__assign( genericComponentOverrideContext.__assign({}, event.sessionContext), { loading: false, invalidClaims: [] } @@ -2446,7 +2462,7 @@ var SessionAuth = function (_a) { } return undefined; }, - [props, setContext, context.loading, userContext, navigate, redirectToLogin] + [props, setContextIfChanged, context.loading, userContext, navigate, redirectToLogin] ); if (props.requireAuth !== false && (context.loading || !context.doesSessionExist)) { return null; diff --git a/lib/ts/recipe/session/sessionAuth.tsx b/lib/ts/recipe/session/sessionAuth.tsx index 807ba4270..3371c5b1b 100644 --- a/lib/ts/recipe/session/sessionAuth.tsx +++ b/lib/ts/recipe/session/sessionAuth.tsx @@ -68,6 +68,22 @@ const SessionAuth: React.FC> = ({ children, // Reusing the parent context was removed because it caused a redirect loop in an edge case // because it'd also reuse the invalid claims part until it loaded. const [context, setContext] = useState({ loading: true }); + const setContextIfChanged = useCallback( + (newValue: SessionContextType) => { + setContext((oldValue) => { + // We can't do this check before re-validation because there are be validators that depend on the current time + // Since the context is constructed by the same functions the property order should be stable, meaning that + // a simple JSON string check should be sufficient. + // Plus since this is just an optimization it is fine to have false positives, + // and this method won't have false negatives (where we'd miss an update). + if (JSON.stringify(oldValue) !== JSON.stringify(newValue)) { + return newValue; + } + return oldValue; + }); + }, + [setContext] + ); const session = useRef(); @@ -191,7 +207,7 @@ const SessionAuth: React.FC> = ({ children, if (failureRedirectInfo.redirectPath !== undefined) { if (validateAndCompareOnFailureRedirectionURLToCurrent(failureRedirectInfo.redirectPath)) { - setContext(toSetContext); + setContextIfChanged(toSetContext); return; } else { return await SuperTokens.getInstanceOrThrow().redirectToUrl( @@ -209,7 +225,7 @@ const SessionAuth: React.FC> = ({ children, message: "Showing access denied screen because a claim validator failed", claimValidationError: failureRedirectInfo.failedClaim, }); - return setContext({ + return setContextIfChanged({ ...toSetContext, accessDeniedValidatorError: failureRedirectInfo.failedClaim, }); @@ -217,7 +233,7 @@ const SessionAuth: React.FC> = ({ children, } } - setContext(toSetContext); + setContextIfChanged(toSetContext); }, [ context.loading, @@ -262,7 +278,7 @@ const SessionAuth: React.FC> = ({ children, if ( validateAndCompareOnFailureRedirectionURLToCurrent(failureRedirectInfo.redirectPath) ) { - setContext({ ...event.sessionContext, loading: false, invalidClaims }); + setContextIfChanged({ ...event.sessionContext, loading: false, invalidClaims }); } else { return await SuperTokens.getInstanceOrThrow().redirectToUrl( failureRedirectInfo.redirectPath, @@ -279,7 +295,7 @@ const SessionAuth: React.FC> = ({ children, message: "Showing access denied screen because a claim validator failed", claimValidationError: failureRedirectInfo.failedClaim, }); - return setContext({ + return setContextIfChanged({ ...event.sessionContext, loading: false, invalidClaims, @@ -287,15 +303,15 @@ const SessionAuth: React.FC> = ({ children, }); } } - setContext({ ...event.sessionContext, loading: false, invalidClaims }); + setContextIfChanged({ ...event.sessionContext, loading: false, invalidClaims }); return; } case "SIGN_OUT": - setContext({ ...event.sessionContext, loading: false, invalidClaims: [] }); + setContextIfChanged({ ...event.sessionContext, loading: false, invalidClaims: [] }); return; case "UNAUTHORISED": - setContext({ ...event.sessionContext, loading: false, invalidClaims: [] }); + setContextIfChanged({ ...event.sessionContext, loading: false, invalidClaims: [] }); if (props.onSessionExpired !== undefined) { props.onSessionExpired(); } else if (props.requireAuth !== false && props.doRedirection !== false) { @@ -316,7 +332,7 @@ const SessionAuth: React.FC> = ({ children, return session.current.addEventListener(onHandleEvent); } return undefined; - }, [props, setContext, context.loading, userContext, navigate, redirectToLogin]); + }, [props, setContextIfChanged, context.loading, userContext, navigate, redirectToLogin]); if (props.requireAuth !== false && (context.loading || !context.doesSessionExist)) { return null; diff --git a/test/end-to-end/signin.test.js b/test/end-to-end/signin.test.js index 877cfe70b..30fd6b2db 100644 --- a/test/end-to-end/signin.test.js +++ b/test/end-to-end/signin.test.js @@ -213,6 +213,32 @@ describe("SuperTokens SignIn", function () { await waitForSTElement(page, "[data-supertokens~=generalError]", true); }); + it("should work visiting no required session page", async function () { + consoleLogs = await clearBrowserCookiesWithoutAffectingConsole(page, consoleLogs); + let cookies = await page.cookies(); + assert.deepStrictEqual(cookies.length, 1); + assert.deepStrictEqual(cookies[0].name, "st-last-access-token-update"); + await page.evaluate(() => window.localStorage.setItem("signoutOnSessionNotExists", "true")); + await Promise.all([ + page.goto(`${TEST_CLIENT_BASE_URL}/dashboard-no-auth`), + page.waitForNavigation({ waitUntil: "networkidle0" }), + ]); + + let text = await getTextInDashboardNoAuth(page); + + assert.strictEqual(text, "Not logged in"); + // We wait for some time to ensure we didn't get into a sign-out loop + await new Promise((res) => setTimeout(res, 1000)); + assert.deepStrictEqual(consoleLogs, [ + "ST_LOGS SESSION OVERRIDE ADD_FETCH_INTERCEPTORS_AND_RETURN_MODIFIED_FETCH", + "ST_LOGS SESSION OVERRIDE ADD_AXIOS_INTERCEPTORS", + "ST_LOGS SESSION OVERRIDE SIGN_OUT", + "ST_LOGS SESSION ON_HANDLE_EVENT SIGN_OUT", + "ST_LOGS SESSION OVERRIDE SIGN_OUT", + "ST_LOGS SESSION ON_HANDLE_EVENT SIGN_OUT", + ]); + }); + it("Successful Sign In with no required session page", async function () { await toggleSignInSignUp(page); await defaultSignUp(page); @@ -492,8 +518,7 @@ describe("SuperTokens SignIn", function () { ), page.waitForNavigation({ waitUntil: "networkidle0" }), ]); - const { pathname, search } = await page.evaluate(() => window.location); - assert.deepStrictEqual(pathname + search, "/redirect-heree?foo=bar"); + await waitForUrl(page, "/redirect-heree?foo=bar", false); }); it("Successful Sign In with redirect to, redirectToPath directly without trailing slash", async function () {