DRF throw django .core.exceptions.ImproperlyConfigured при переопределении get_queryset - PullRequest
4 голосов
/ 13 февраля 2020

Django Rest Framework throw:

django .core.exceptions.ImproperlyConfigured: Не удалось разрешить URL-адрес для связи с гиперссылкой, используя имя представления «customuser-detail». Возможно, вам не удалось включить соответствующую модель в API или неправильно настроить атрибут lookup_field в этом поле.

при попытке переопределить get_queryset . Сериализатор My User:


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 Ответ

1 голос
/ 13 февраля 2020

Согласно документации , basename, который вы предоставляете при регистрации ViewSet на маршрутизаторе, будет использоваться для генерации шаблонов URL. Вы указали basename=CustomUser, который имеет заглавные буквы.

Как указывает на ошибку (также: см. документация о том, как определяются гиперссылки и создаются имена по умолчанию), ваше базовое имя должно быть в нижнем регистре.

Если вы хотите сохранить заглавные буквы, см. Предыдущую ссылку на документацию для реализации пользовательского view_name='CustomUser-detail' на HyperlinkedModelSerializer.

...