Next.js Discord

Discord Forum

Is fetching and redirecting in layout a bad pattern?

Answered
Yongho Lee posted this in #help-forum
Open in Discord
I wonder is it a bad pattern to fetch and redirect in ⁠layout. Redirection in ⁠layout means you don't want the user to view the ⁠page. However, I concern about that ⁠layout and ⁠page work simultaneously.

Let's take an example. We need to create several admin pages and block general users from accessing those pages. So, where should the logic for blocking access be located? What about locating this at layout? At first, the intention to call ⁠checkAdmin in layout is to reduce unnecessary fetching in client routing from /admin to /admin/other. I thought check again is unnecessary in client routing under the same layout.

// /app/admin/layout.js
export default async function Layout() {
    const isAdmin = await checkAdmin()

    if (!isAdmin) redirect('/home')

    return <>{children}</>
} 

// /app/admin/page.js
export default async function Page() {
    const credentialData = await fetchApiForAdming()

    return <>{children}</>
}

// /app/admin/other/page.js
export default async function Page() {
    const credentialData = await fetchApiForAdming()

    return <>{children}</>
}


At first glance, it's a quite reasonable choice. Let's go back to the problem that I mentioned at first. In the above code, fetchApiForAdming is called regardless of whether the user is an admin or not. Of course, RSC doesn't return ⁠the credential data and only redirects to ⁠home even if ⁠fetchApiForAdmin throws an error. However, is it appropriate for ⁠fetchApiForAdmin to be called even for requests from regular users, even though it happens server-side and is not transmitted to the client?

Another approach would be to send this logic to middleware, but it is concerning because middleware runs in a separate environment from the app, Edge Runtime, which means caching is not possible.
Answered by B33fb0n3
yea, your redirect won't be efficent, when it's depending on the auth of the user. Imagine the user bypasses this check, like I mentioned, then there is no effective redirect. In other words it means: redirect in layout is fine, but not when it's depending on auth
View full answer

5 Replies

@Yongho Lee I wonder is it a bad pattern to fetch and redirect in ⁠layout. Redirection in ⁠layout means you don't want the user to view the ⁠page. However, I concern about that ⁠layout and ⁠page work simultaneously. Let's take an example. We need to create several admin pages and block general users from accessing those pages. So, where should the logic for blocking access be located? What about locating this at `layout`? At first, the intention to call ⁠`checkAdmin` in layout is to reduce unnecessary fetching in client routing from /admin to /admin/other. I thought check again is unnecessary in client routing under the same `layout`. ts // /app/admin/layout.js export default async function Layout() { const isAdmin = await checkAdmin() if (!isAdmin) redirect('/home') return <>{children}</> } // /app/admin/page.js export default async function Page() { const credentialData = await fetchApiForAdming() return <>{children}</> } // /app/admin/other/page.js export default async function Page() { const credentialData = await fetchApiForAdming() return <>{children}</> } At first glance, it's a quite reasonable choice. Let's go back to the problem that I mentioned at first. In the above code, `fetchApiForAdming` is called regardless of whether the user is an admin or not. Of course, RSC doesn't return ⁠the credential data and only redirects to ⁠home even if ⁠`fetchApiForAdmin` throws an error. However, is it appropriate for ⁠`fetchApiForAdmin` to be called even for requests from regular users, even though it happens server-side and is not transmitted to the client? Another approach would be to send this logic to `middleware`, but it is concerning because `middleware` runs in a separate environment from the app, Edge Runtime, which means caching is not possible.
yea, when checking your auth inside your layout, the user is able to bypass this check. And that mean for you: a security vulnability. Don't check the auth inside your layout. You can read about it here ([click me](https://github.com/eric-burel/securing-rsc-layout-leak)).

Instead check in:
1. Middleware or
2. Page or
3. Function
Mallow bee
Check auth right before every action you want to guard, whether that's writing to the DB or calling an internal API. If you ever add a permissions model you'll have to do this anyway

then the route level redirects are just a UX enhancement to stop people from performing actions that would throw errors
Thanks comments @B33fb0n3 @Mallow bee
My concern would rather relate to efficiency than security because the ⁠fetchApiForAdmin function also restricts general users and throws an error if so. I didn't even want to call ⁠fetchApiForAdmin. As you mentioned, I think redirecting in layout was not a good idea.
@Yongho Lee Thanks comments <@301376057326567425> <@231856762750369793> My concern would rather relate to efficiency than security because the ⁠`fetchApiForAdmin` function also restricts general users and throws an error if so. I didn't even want to call ⁠`fetchApiForAdmin`. As you mentioned, I think redirecting in layout was not a good idea.
yea, your redirect won't be efficent, when it's depending on the auth of the user. Imagine the user bypasses this check, like I mentioned, then there is no effective redirect. In other words it means: redirect in layout is fine, but not when it's depending on auth
Answer