最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

Laravel Sanctum SPA Authentication Not Working with Next.js Frontend - Stack Overflow

programmeradmin1浏览0评论

Problem

I'm building an application with a Next.js frontend and Laravel backend using Sanctum for SPA authentication. When I login from the Next.js app, I can see the authentication cookies being set in the browser, but subsequent requests to get user information return "unauthenticated" errors. NOTE: Everything is working in Postman correctly.

Environment

  • Frontend: Next.js
  • Backend: Laravel with Sanctum
  • Authentication: Laravel Sanctum SPA Authentication
  • Everything works correctly when testing in Postman

What I've Tried

The login process works fine, and I can confirm cookies are set in the browser after login. However, when I try to fetch the authenticated user information, I get "unauthorized" errors. I tried GET request at "/api/v1/user" in Postman and it works.

Code

My login function:

const baseUrl = process.env.NEXT_PUBLIC_BASE_URL;

async function fetchCsrfCookie() {
    await fetch(`${process.env.NEXT_PUBLIC_BASE_URL}/sanctum/csrf-cookie`, {
        method: 'GET',
        credentials: 'include',
    });
}
export const fetchSignIn = async (username: string, password: string) => {
    //   1) Init Sanctum
    await fetchCsrfCookie();

    // 2) Login
    const res = await fetch(`${baseUrl}/api/v1/login`, {
        method: 'POST',
        credentials: 'include',
        headers: { 'Content-Type': 'application/json' },
        body: JSON.stringify({ username, password, remember: true }),
    });
    if (!res.ok) {
        throw new Error('Login failed');
    }
    return res.json();
};

Fetch user function:

export const fetchAuthUser = async () => {
    try {
        const res = await fetch(`${process.env.NEXT_PUBLIC_BASE_URL}/api/v1/user`, {
            method: 'GET',
            credentials: 'include',
            headers: {
                'Content-Type': 'application/json',
            },
        });

        if (!res.ok) {
            const errorData: any = await res.json();
            console.error('Error data:', errorData);
            throw new Error(errorData.message || res.statusText);
        }

        return res.json();
    } catch (error) {
        console.error(`Failed to fetch resource:`, error);
        throw error;
    }
};

Problem

I'm building an application with a Next.js frontend and Laravel backend using Sanctum for SPA authentication. When I login from the Next.js app, I can see the authentication cookies being set in the browser, but subsequent requests to get user information return "unauthenticated" errors. NOTE: Everything is working in Postman correctly.

Environment

  • Frontend: Next.js
  • Backend: Laravel with Sanctum
  • Authentication: Laravel Sanctum SPA Authentication
  • Everything works correctly when testing in Postman

What I've Tried

The login process works fine, and I can confirm cookies are set in the browser after login. However, when I try to fetch the authenticated user information, I get "unauthorized" errors. I tried GET request at "/api/v1/user" in Postman and it works.

Code

My login function:

const baseUrl = process.env.NEXT_PUBLIC_BASE_URL;

async function fetchCsrfCookie() {
    await fetch(`${process.env.NEXT_PUBLIC_BASE_URL}/sanctum/csrf-cookie`, {
        method: 'GET',
        credentials: 'include',
    });
}
export const fetchSignIn = async (username: string, password: string) => {
    //   1) Init Sanctum
    await fetchCsrfCookie();

    // 2) Login
    const res = await fetch(`${baseUrl}/api/v1/login`, {
        method: 'POST',
        credentials: 'include',
        headers: { 'Content-Type': 'application/json' },
        body: JSON.stringify({ username, password, remember: true }),
    });
    if (!res.ok) {
        throw new Error('Login failed');
    }
    return res.json();
};

Fetch user function:

export const fetchAuthUser = async () => {
    try {
        const res = await fetch(`${process.env.NEXT_PUBLIC_BASE_URL}/api/v1/user`, {
            method: 'GET',
            credentials: 'include',
            headers: {
                'Content-Type': 'application/json',
            },
        });

        if (!res.ok) {
            const errorData: any = await res.json();
            console.error('Error data:', errorData);
            throw new Error(errorData.message || res.statusText);
        }

        return res.json();
    } catch (error) {
        console.error(`Failed to fetch resource:`, error);
        throw error;
    }
};
Share Improve this question asked Apr 2 at 3:50 Marko TicinovicMarko Ticinovic 11 bronze badge New contributor Marko Ticinovic is a new contributor to this site. Take care in asking for clarification, commenting, and answering. Check out our Code of Conduct. 8
  • Sorry, can you share your laravel side's .env? I would like to see your session configurations. Please do remove any secret env value such as db password. – Tatachiblob Commented Apr 2 at 3:54
  • in your fetchAuthUser fetch, unless you are explicitly set in your fetch that you added your JWT token, then it will not include it when submitting the request. – Mr. Kenneth Commented Apr 2 at 7:12
  • SESSION_DRIVER=redis SESSION_LIFETIME=120 SESSION_ENCRYPT=true SESSION_PATH=/ SESSION_DOMAIN="localhost" SESSION_SECURE_COOKIE=false SANCTUM_STATEFUL_DOMAINS="localhost:3000" @Tatachiblob – Marko Ticinovic Commented yesterday
  • Thanks, can you try to change this env variable? SESSION_DOMAIN=null – Tatachiblob Commented yesterday
  • 1 @Tatachiblob I just found a solution. Problem was in SANCTUM_STATEFUL_DOMAINS="localhost:3000" . Next is runned at port localhost 3000 by default, but when i start backend, port 3000 is taken and Next is running at different port( 3002 ) in my case. I updated the .env and it works. Thank you. – Marko Ticinovic Commented yesterday
 |  Show 3 more comments

1 Answer 1

Reset to default 0

I figured out the issue! The problem was in SANCTUM_STATEFUL_DOMAINS="localhost:3000" . When I run backend it takes 3000 port and my NEXT server is running at different port(3002) in my case. I updated .env to 3002 port and it worked.

发布评论

评论列表(0)

  1. 暂无评论