问题描述
我想通过 SimpleJWT
cookie 发送 access
refresh
和 HttpOnly
令牌。我已经定制了索赔。我在设置 cookie 的 post()
中定义了一个 MyObtainTokenPairView(TokenObtainPairView)
方法。这是我的代码:
from .models import CustomUser
class MyObtainTokenPairView(TokenObtainPairView):
permission_classes = (permissions.AllowAny,)
serializer_class = MyTokenObtainPairserializer
def post(self,request,*args,**kwargs):
serializer = self.serializer_class()
response = Response()
tokens = serializer.get_token(CustomUser)
access = tokens.access
response.set_cookie('token',access,httponly=True)
return response
它返回此错误:
AttributeError: 'RefreshToken' object has no attribute 'access'
序列化器:
class MyTokenObtainPairserializer(TokenObtainPairserializer):
@classmethod
def get_token(cls,user):
print(type(user))
token = super().get_token(user)
token['email'] = user.email
return token
但它只是不起作用。我认为我不应该像这样在这里定义 post()
方法。我想如果我只能在序列化程序中返回 get_token()
函数的值,我可以将它设置为 HttpOnly
cookie。但是,我不知道该怎么做。
如何在 access
cookie 中设置 refresh
和 HttpOnly
令牌?
编辑: 我根据 aNowlinorbit 的回答进行了这些更改:
我把我的序列化器改成了这样:
class MyTokenObtainPairserializer(TokenObtainPairserializer):
def validate(self,attrs):
attrs = super().validate(attrs)
token = self.get_token(self.user)
token["email"] = self.user.email
return token
由于此令牌默认包含刷新令牌,因此我决定仅返回此 token
将同时提供 access
和 refresh
令牌。
如果我添加类似 token["access"] = str(token.access_token)
的任何内容,它只会在它已经包含的刷新令牌字符串中添加访问令牌字符串。
但再次在视图中,我找不到如何获取刷新令牌。我无法使用 serializer.validated_data.get('refresh',None)
获取它,因为现在我要从包含所有内容的序列化程序返回 token
。
我改变了我的看法:
class MyObtainTokenPairView(TokenObtainPairView):
permission_classes = (permissions.AllowAny,**kwargs):
response = super().post(request,**kwargs)
response.set_cookie('token',token,httponly=True)
return response
现在说:
NameError: name 'token' is not defined
这里有什么问题?在视图中,我想获取从序列化程序返回的 token
,然后使用 token.access_token
获取访问令牌并将 refresh
和 access
设置为 cookie。
解决方法
我会不理会.get_token()
,而是专注于.validate()
。在您的 MyTokenObtainPairSerializer
中,我将删除您对 .get_token()
的更改并添加以下内容
def validate(self,attrs):
data = super().validate(attrs)
refresh = self.get_token(self.user)
data["refresh"] = str(refresh) # comment out if you don't want this
data["access"] = str(refresh.access_token)
data["email"] = self.user.email
""" Add extra responses here should you wish
data["userid"] = self.user.id
data["my_favourite_bird"] = "Jack Snipe"
"""
return data
通过使用 .validate()
方法,您可以选择希望从序列化程序对象的 validated_data
属性返回哪些数据。 注意我还在序列化程序返回的数据中包含了刷新令牌。同时拥有刷新和访问令牌很重要。如果用户没有刷新令牌,他们将不得不在访问令牌过期时再次登录。刷新令牌使他们无需再次登录即可获得新的访问令牌。
如果出于某种原因您不想要刷新令牌,请将其从 validate()
序列化程序方法中删除并相应地调整视图。
在这个 post 方法中,我们验证序列化程序并访问其验证数据。
def post(self,request,*args,**kwargs):
# you need to instantiate the serializer with the request data
serializer = self.serializer(data=request.data)
# you must call .is_valid() before accessing validated_data
serializer.is_valid(raise_exception=True)
# get access and refresh tokens to do what you like with
access = serializer.validated_data.get("access",None)
refresh = serializer.validated_data.get("refresh",None)
email = serializer.validated_data.get("email",None)
# build your response and set cookie
if access is not None:
response = Response({"access": access,"refresh": refresh,"email": email},status=200)
response.set_cookie('token',access,httponly=True)
response.set_cookie('refresh',refresh,httponly=True)
response.set_cookie('email',email,httponly=True)
return response
return Response({"Error": "Something went wrong",status=400)
如果您不想要刷新令牌,您可以删除以 refresh =
开头的行并删除添加 refresh
cookie 的行。