覆盖get_queryset时DRF抛出django.core.exceptions.ImproperlyConfigure

时间:2020-02-13 15:56:17

标签: python django django-rest-framework

Django Rest Framework抛出:

django.core.exceptions.ImproperlyConfigured:无法使用视图名称“ customuser-detail”解析超链接关系的URL。您可能没有在API中包含相关模型,或者在此字段上错误地配置了lookup_field属性。

当我尝试 覆盖get_queryset 时。 我的用户序列化器:


class UserSerializer(serializers.HyperlinkedModelSerializer):
    """
    Represent User Serializer class.
    """
    teacher_account = TeacherSerializer(required=False)
    student_account = StudentSerializer(required=False)

    account_type = serializers.IntegerField(required=True)

    class Meta:
        model = CustomUser
        fields = ['url', 'username', "password", 'email', 'first_name', 'last_name', "account_type", 'teacher_account', 'student_account']

        email_validator = UniqueValidator(queryset=CustomUser.objects.all(), message="A user with that email already exists.")

        extra_kwargs = {
            "password": {"write_only": True},
            "email": {"required": True, "validators": [email_validator]}
        }

    @staticmethod
    def setup_eager_loading(queryset):
        queryset = queryset.select_related('teacher_account', 'student_account')
        return queryset

users / models.py:

class StudentAccount(models.Model):
    """
    Represent student's account model.

    """
    classes = models.ManyToManyField('classroom.Class', related_name="students")


class TeacherAccount(models.Model):
    """
    Represent teacher's account.

    use get_subject_name from {root}/utils.py for get name of the subject.
    """

    subject_id = models.PositiveSmallIntegerField("Предмет", choices=SUBJECTS_CHOICES, blank=False, default=0)


class CustomUser(AbstractUser):
    """
    Represent Custom user model, inherited from AbstractUser

    account_type = 0(teacher) or 1(student)
    """
    student_account = models.OneToOneField(StudentAccount, on_delete=models.CASCADE, blank=True, null=True, related_name="user")
    teacher_account = models.OneToOneField(TeacherAccount, on_delete=models.CASCADE, blank=True, null=True, related_name="user")

    account_type = models.PositiveSmallIntegerField(default=1)

    first_name = models.CharField("-", max_length=30, blank=False)
    last_name = models.CharField("-", max_length=150, blank=False)

我的views.py:

class UserViewSet(viewsets.ModelViewSet):
    """
    API endpoint that allows users to be viewed or edited.
    """

    permission_classes = (permissions.AllowAny,)
    # queryset = CustomUser.objects.all()
    # queryset_raw = CustomUser.objects.all()
    # queryset = UserSerializer.setup_eager_loading(queryset_raw)
    model = CustomUser
    serializer_class = UserSerializer

    def get_queryset(self):
        queryset = CustomUser.objects.all()
        # queryset = self.get_serializer_class().setup_eager_loading(queryset)
        return queryset

我的urls.py:

from django.urls import path, include

from rest_framework import routers
from knox import views as knox_views

from .views import UserViewSet, ClassViewSet, LoginView, LessonViewSet

router = routers.DefaultRouter()
router.register('users', UserViewSet, basename="CustomUser")
router.register('classes', ClassViewSet)
router.register('lessons', LessonViewSet)


urlpatterns = [
    path('login/', LoginView.as_view(), name='knox-login'),
    path('logout/', knox_views.LogoutView.as_view(), name="knox_logout"),
    path('logoutall/', knox_views.LogoutAllView.as_view(), name="knox_logoutall"),
    path('', include(router.urls)),
    path('api-auth/', include('rest_framework.urls', namespace='rest_framework'))
]

我意识到,如果我将覆盖显示用户模型详细信息的页面的url覆盖,则可以解决此问题。但是我认为还有更正确的答案。

path('api/users/<int:pk>/', UserViewSet.as_view({"get": "retrieve", "put": "update", "delete": "destroy"}), name="customuser-detail")

1 个答案:

答案 0 :(得分:1)

根据documentation,将basename注册到路由器时提供的ViewSet将用于生成URL模式。您提供了basename=CustomUser,该字母大写。

如错误所指出(另请参见:关于如何确定超链接视图和如何构造默认名称的信息,请参见documentation),您的基本名称应为小写。

如果您想保留大写字母,请参阅以前的文档链接,以在view_name='CustomUser-detail'上实现自定义HyperlinkedModelSerializer

相关问题