如何覆盖 Djoser 基本端点用户/我

Posted

技术标签:

【中文标题】如何覆盖 Djoser 基本端点用户/我【英文标题】:How Override Djoser base endpoint users/me 【发布时间】:2021-06-17 11:55:25 【问题描述】:

我对 Djoser 有点陌生,我想做的是当经过身份验证的用户导航到 users/me API 端点时,它会返回用户 ID 电子邮件和名称。我希望它也返回自定义信息,我该怎么做?

【问题讨论】:

【参考方案1】:

Djoser 只是一个 Django 应用程序。 只需用您自己的类覆盖 Djoser 的类,并将您的身份验证 URL 指向那些而不是包的那些。 我不得不这样做,这是我所做的示例。

from djoser.views import TokenDestroyView as DjoserTokenDestroyView
from djoser.views import UserViewSet

class CustomUserViewSet(UserViewSet):
    """
    Overriding djoser's Users view
    """

    @extend_schema(summary="Me", responses=200: LoginSerializer, 404: UserNotFoundResponse, 401: FailedLogin)
    @action(["get", "put"], detail=False)
    def me(self, request, *args, **kwargs):
        """
        get the current User's data and KPIs settings
        """
        if request.method == MethodsNames.GET.upper():
            user = User.objects.filter(email=self.get_instance()).select_related(ACCOUNT).prefetch_related(
                RelatedNames.AUTH_TOKEN,
                'account__bundles', 'account__bundles__bundle_kpis',
                RelatedNames.USER_SELECTED_KPIS)[0]
            user_data = LoginSerializer(user, many=False)
            return Response(user_data.data)
       # some more overriding here...

然后我的 urls 文件看起来像:

from django.conf.urls import url
from django.urls import include, path
from rest_framework.routers import DefaultRouter

from users.views import TokenCreateView, CustomUserViewSet, TokenDestroyView

users = DefaultRouter()
users.register("users", CustomUserViewSet, basename='users')

login = DefaultRouter()
login.register('token', TokenCreateView, basename='login')

urlpatterns = [
    path('', include(login.urls)),
    path('', include(users.urls)),
    url(r"^token/logout/?$", TokenDestroyView.as_view(), name="logout"),
]

而不是(来自 Djoser 文档):

urlpatterns = [
    (...),
    url(r'^auth/', include('djoser.urls')),
]

【讨论】:

嘿@yovel cohen 你能看看我的另一个问题吗? ***.com/questions/66722096/… 好的,如果答案对您有帮助,您可以投票吗? @ElieFrancois 如果不包含 url,使用 DJOSER 有什么意义?据我了解,如果您不将 djoser.urls 包含到您的应用程序中,那么 djoser 的所有端点都无法通过您的 API 访问,对吧?因此不能使用例如 djoser 注册或 password_reset 功能。那么使用 djoser 有什么意义呢?如果您不使用他们的网址,它如何派上用场?【参考方案2】:

虽然这可能只能回答您问题的一半,但如果您使用自定义用户,那么您需要添加 REQUIRED_FIELDS = ['field_you_want_included']

例如,这就是我的一个自定义用户配置文件的样子。

class CustomUser(AbstractUser):
    username = None
    user_id = models.UUIDField(default=uuid4, primary_key=True, editable=False)
    email = models.EmailField(_('email address'), unique=True)
    receive_news = models.BooleanField(null=True, blank=True, default=False)
    birth_date = models.DateField(null=True, blank=True)
    is_premium = models.BooleanField(default=True)

    USERNAME_FIELD = 'email'
    # FIELDS HERE WILL BE ADDED TO USERS/ME ENDPOINT
    REQUIRED_FIELDS = ['is_premium']
    objects = UserAccountManager()

如果有很多关于使用 djoser 制作自定义用户的信息,那么这里重要的是 required_fields 列表。现在,当访问用户/我时,它还将包括“is_premium”。 但是,对于更复杂的包含,我会尝试覆盖视图和序列化程序。我自己没有尝试过,但如果我这样做了,我会编辑这篇文章。

【讨论】:

以上是关于如何覆盖 Djoser 基本端点用户/我的主要内容,如果未能解决你的问题,请参考以下文章

如何从 djoser 端点(.../users/me)检索用户数据

自定义 djoser 创建用户端点

如何在 Djoser 中禁用用户名更改?

DRF:Djoser 覆盖自定义用户序列化程序

在 Djoser 和 Rest Framework 中使用自定义字段注册用户

Spring boot:如何跳过特定端点的基本身份验证