首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >Laravel-vue jwt令牌刷新

Laravel-vue jwt令牌刷新
EN

Stack Overflow用户
提问于 2018-10-19 09:41:11
回答 1查看 3.6K关注 0票数 1

我正在使用来自Tymon的jwtAuth package来处理从我的laravel后端到vue spa前端的身份验证,我正在创建AuthController,我从文档中获取了很大一部分,并对其进行了一些调整以满足我的需求。从登录到注销,再到令牌过期,一切都很正常。

问题是,我确实看到控制器上有一个令牌刷新功能,如果我的猜测是正确的,它将刷新客户端已经拥有的当前令牌。但如何做到这一点呢?如何在我前端处理刷新标记?因为每隔60分钟(默认的令牌生存期)就会抛出401,这很让人恼火。

我想要的是,也许每次用户请求后端时,它都会刷新令牌或增加令牌的生命周期。因此,仅当用户在整个60分钟内处于空闲状态时,令牌才会过期。

我们能做到吗?这是最佳实践吗?我对整个jwt和令牌的事情都是新手,在过去,我只依赖于laravel令牌过期,因为我没有使用spa,但刀片前端,所以大多数情况下不需要扰乱laravel身份验证用户的方式。

因为这里添加的信息是我认为与整个auth相关的每个文件。

这是我的authcontroller

代码语言:javascript
复制
<?php

namespace App\Http\Controllers;

use DB;
use Hash;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\Validator;
use Illuminate\Support\Facades\Input;
use App\Http\Controllers\Controller;
use App\User;
use Response;

class Authcontroller extends Controller
{
    /**
     * Create a new AuthController instance.
     *
     * @return void
     */
    public function __construct()
    {
        $this->middleware('auth:api', ['except' => ['login']]);
    }

    /**
     * Get a JWT via given credentials.
     *
     * @return \Illuminate\Http\JsonResponse
     */
    public function login()
    {
        $credentials = request(['username', 'password']);

        if (! $token = auth('api')->attempt($credentials)) {
            return response()->json(['error' => 'Unauthorized'], 401);
        }

        return $this->respondWithToken($token);
    }

    /**
     * Get the authenticated User.
     *
     * @return \Illuminate\Http\JsonResponse
     */
    public function me()
    {
        return response()->json(auth('api')->user());
    }

    /**
     * Log the user out (Invalidate the token).
     *
     * @return \Illuminate\Http\JsonResponse
     */
    public function logout()
    {
        auth('api')->logout();

        return response()->json(['message' => 'Successfully logged out']);
    }

    /**
     * Refresh a token.
     *
     * @return \Illuminate\Http\JsonResponse
     */
    public function refresh()
    {
        return $this->respondWithToken(auth('api')->refresh());
    }

    /**
     * Get the token array structure.
     *
     * @param  string $token
     *
     * @return \Illuminate\Http\JsonResponse
     */
    protected function respondWithToken($token)
    {
        $id = auth('api')->user()->getId();
        $kelas = User::with('pus','cu')->findOrFail($id);

        return response()->json([
            'access_token' => $token,
            'user' => $kelas,
            'token_type' => 'bearer',
            'expires_in' => auth('api')->factory()->getTTL() * 60
        ]);
    }

    public function guard()
    {
        return Auth::Guard('api');
    }

}

这是我的API路由

代码语言:javascript
复制
Route::group(['prefix' => 'auth'],function($router){
    Route::post('/login', 'AuthController@login');
    Route::post('/logout', 'AuthController@logout');
    Route::post('/refresh', 'AuthController@refresh');
    Route::get('/me', 'AuthController@me');
});

下面是我的vue general.js文件,它处理路由并将报头提供给Axios

代码语言:javascript
复制
export function initialize(store, router) {
  router.beforeEach((to, from, next) => {
      const requiresAuth = to.matched.some(record => record.meta.requiresAuth);
      const currentUser = store.state.auth.currentUser;

      if(requiresAuth && !currentUser) {
          next('/login');
      } else if(to.path == '/login' && currentUser) {
          next('/');
      } else {
          next();
      }
  });

  axios.interceptors.response.use(null, (error) => {
      if (error.response.status == 401) {
          store.dispatch('auth/logout');
          router.push('/login');
      }

      return Promise.reject(error);
  });

  if (store.state.auth.currentUser) {
      setAuthorization(store.state.auth.currentUser.token);
  }
}

export function setAuthorization(token) {
  axios.defaults.headers.common["Authorization"] = `Bearer ${token}`
}

下面是处理登录的auth.js

代码语言:javascript
复制
import { setAuthorization } from "./general";

export function login(credentials){
  return new Promise((res,rej) => {
    axios.post('/api/auth/login', credentials)
      .then((response) => {
        setAuthorization(response.data.access_token);
        res(response.data);
      })
      .catch((err) => {
        rej("Username atau password salah");
      })
  })
}
EN

回答 1

Stack Overflow用户

发布于 2018-10-19 12:39:46

您可以调整令牌过期时间(从.env设置为JWT_TTL)和刷新时间(JWT_REFRESH_TTL)以满足您的需要。并检查令牌是否有效和/或是否需要在中间件中刷新令牌,以便在需要时尽快刷新令牌。

至于这是否是一个好的实践,请参阅您的Laravel项目的config/jwt.php中对JWT_REFRESH_TTL的评论。

对我来说运行良好的解决方案是使用扩展Tymon\JWTAuth\Http\Middleware\BaseMiddleware的自定义中间件。样板文件看起来像这样:

代码语言:javascript
复制
Class TryTokenRefresh extends BaseMiddleware
{
    public function handle($request, Closure $next)
    {
        $newToken = $this->tryRefresh($request);
        if ($newToken) {
           // in case there's anything further to be done with the token
           // we want that code to have a valid one
           $request->headers->set('Authorization', 'Bearer ' . $newToken);
        }

        ...
        ...
        $response = $next($request);
        ...
        ...

        if ($newToken) {
            // send new token back to frontend
            $response->headers->set('Authorization', $newToken);
        }

        return $response;
    }

    // Refresh the token
    protected function tryRefresh()
    {
        try {
            $token = $this->auth->parseToken()->refresh();
            return $token;
        } catch (JWTException $e) {
            // token expired? force logout on frontend
            throw new AuthenticationException();
        }

        return null;
    }

在前端,它就像在响应中查找Authorization头一样简单:

代码语言:javascript
复制
// check for the `Authorization` header in each response - refresh on frontend if found
axios.interceptors.response.use((response) => {
  let headers = response.headers

  // your 401 check here
  // token refresh - update client session
  if (headers.authorization !== undefined) {
   setAuthorization(headers.authorization);
  }

  return response
})

希望这能有所帮助。

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/52884747

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档