Evgenii Legotckoi
Evgenii Legotckoi26 апреля 2020 г. 15:08

Django - Урок 052. Переопределение модели пользователя

Существуют различные варианты работы с моделью пользователя для создания профиля пользователя и добавления определенной информации, например, фотографии профиля. Один из способов — добавить модель профиля с полем OneToOneField для пользователя. Этот вариант проще, чем прокси-модель или переопределение пользовательской модели. Но лично для меня переопределение пользовательской модели оказалось более эффективным с точки зрения организации кода и написания различных методов, позволяющих извлекать нужную информацию из пользовательского объекта в шаблонах, не прибегая к написанию шаблонных тегов.

Поэтому рассмотрим именно вариант переопределения модели пользователя в проекте на Django


Вступление

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

Я сам делал переопределение пользовательской модели уже когда сайт просуществовал больше двух лет. Однако у меня процесс переопределения прошел относительно гладко, по крайней мере я не помню, чтобы были какие-то проблемы с тем, что я переопределял пользовательскую модель.

Переопределение пользовательской модели

Чтобы правильно переопределить модель пользователя, вам нужно импортировать абстрактную модель пользователя AbstractUser и наследоваться от нее в приложении, где у вас будет модель.

В моем приложении evileg_auth я сделал так

# -*- coding: utf-8 -*-

from django.contrib.auth.models import AbstractUser


class User(AbstractUser):
    pass

Далее вам нужно указать эту модель как новую пользовательскую модель в файле настроек settings.py

AUTH_USER_MODEL = 'evileg_auth.User'

Это нужно для того, чтобы стороннее приложение начало использовать вашу пользовательскую модель вместо стандартной.

Затем вы можете создать миграции и перенести проект

python manage.py makemigrations
python manage.py migrate

Также не забудьте зарегистрировать свою модель в админке

# -*- coding: utf-8 -*-

from django.contrib import admin
from django.contrib.auth.admin import UserAdmin

from .models import User

admin.site.register(User, UserAdmin)

Примечания

Если вы внедряете переопределенную пользовательскую модель в существующий проект, я рекомендую вам выполнить следующие шаги, чтобы переопределенная модель использовалась везде в вашем проекте.

настройки.AUTH_USER_MODEL

Во всех файлах models.py используйте переменную settings.AUTH_USER_MODEL из настроек вместо стандартной модели.

# -*- coding: utf-8 -*-

from django.conf import settings
from django.db import models


class SomeModel(models.Model):
    user = models.ForeignKey(settings.AUTH_USER_MODEL, verbose_name='User')

get_user_model()

Везде в коде, где предполагается, что экземпляр Django уже запущен, используйте динамическое получение класса пользовательской модели через get_user_model()

Например, у меня есть такой миксин

class EInUserProfileMixin:
    user_profile = None

    def dispatch(self, request, *args, **kwargs):
        self.user_profile = get_object_or_404(get_user_model(), username=kwargs['user'], is_active=True)
        return super().dispatch(request, *args, **kwargs)

Стороннее приложение

Если вы пишете стороннее приложение, то рекомендации из раздела заметок должны быть для вас обязательными, потому что иначе ваше стороннее приложение будет бесполезным для тех разработчиков, которые переопределяют пользовательские модели в своих проектах на Django

Преимущества переопределения пользовательской модели

Переопределение модели пользователя имеет, на мой взгляд, большие преимущества по сравнению с использованием OneToOneField, так как позволяет без костылей переопределить диспетчер объектов, а также добавить ряд методов, которые могут пригодиться в вашем проекте. А также эти методы можно вызывать в шаблонах без написания тегов шаблона.

Вот пример части пользовательской модели в моем проекте

class User(AbstractUser):

    last_online = models.DateTimeField(blank=True, null=True)

    objects = UserManager()

    class Meta:
        ordering = ['username']

    def get_absolute_url(self):
        return reverse('users:profile', kwargs={'user': self.username})

Как минимум, мне всегда не хватало использования стандартного метода get_absolute_url для профиля пользователя на сайте.

Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.

Вам это нравится? Поделитесь в социальных сетях!

Владислав Меленчук
  • 27 апреля 2020 г. 7:43

Полезный урок, весьма лучше, чем создать профиль. А как всё это объединить в админке? admin.site.register(User, UserAdmin) - UserAdmin убрал мое созданное поле для абстрактного поля.

Я немного не понял, что именно вы хотите объединить? Вы хотите, чтобы группы и пользователи были в одном groupbox'e?
Или хотите видеть абстрактное поле? Если абстрактное поле, то наследуйтесь от UserAdmin и переопределите переменные fields и т.д., как если бы наследовались от ModelAdmin

Я уже нашел в вашем другом уроке как добавить поле аватар к пользователям. Именно то что вы написали) Спасяб:3
Да ещё бы хотелось чтоб группы и пользователи были в одной группе.

Да ещё бы хотелось чтоб группы и пользователи были в одной группе.

Насчёт такого никогда не заморачивался, мне это не мешало. Но вообще для этого можно использовать прокси модель.

Думаю, что это может выглядеть так

class ProxyGroups(Groups):
    class Meta:
        proxy = True    
        # Если определить ProxyGroup внутри my_auth/models.py,
        # то его app_label будет установлен как 'my_auth' автоматически.
        # Или можно переопределить его так           
        # app_label = 'my_auth'

        # также можно установить некоторые параметры из существующего Group
        # verbose_name = Groups._meta.verbose_name
        # verbose_name_plural = Groups._meta.verbose_name_plural


# in admin.py
admin.site.register(ProxyGroups)

Суть в том, что проксируем стандартную модель Groups в другое приложение, чтобы они отображались в одном app

Благодарю!

Z M
  • 4 мая 2020 г. 5:56

Подскажите, эта серия уроков для django 2.0?

Evgenii Legotckoi
  • 4 мая 2020 г. 5:59
  • (ред.)

Я ещё не переходил на Django 3, поэтому на данный момент все примеры пишутся с использованием Django 2, наиболее старые с использованием Django 1.

Но в принципе, большая часть из статей должна работать и с Django 3. Хотя и не все.

Z M
  • 4 мая 2020 г. 6:02

Понял, большое спасибо! Буду ждать перехода на 3 версию :)

Evgenii Legotckoi
  • 4 мая 2020 г. 6:04

Я тоже его жду. Вернее жду, когда все разработчики третьесторонних библиотек добавят поддержку третьей версии.

t
  • 23 января 2021 г. 8:28

Добрый день, Я вот написал в admin.py
admin.site.register(User, UserAdmin)
Но у меня не появилось новые поля в админке?

Evgenii Legotckoi
  • 31 января 2021 г. 16:30

Если у вашего пользователя нет прав но чтение или изменение модели пользователя, то вы можете и не увидеть эту модель в админке. В этом случае лучше под суперпользователем проверять всё.

t
  • 23 февраля 2021 г. 7:11

А как дать ему эти права?

t
  • 24 февраля 2021 г. 2:56

В данном случае я заходил под superuser но все равно не появлялись эти поля

N
  • 19 апреля 2022 г. 3:22

def initials(self):
return get_initials(self)

В последнем листинге что такое get_initials(self), пайчарм подчеркивает и предлагает создать функцию get_initials

Evgenii Legotckoi
  • 19 апреля 2022 г. 3:58

Лишний кусок кода (Метод для получения инициалов), удалите его.

Комментарии

Только авторизованные пользователи могут публиковать комментарии.
Пожалуйста, авторизуйтесь или зарегистрируйтесь
ОН

C++ - Тест 006. Перечисления

  • Результат:10баллов,
  • Очки рейтинга-10
K
  • KiRi4
  • 7 сентября 2023 г. 17:57

C++ - Тест 002. Константы

  • Результат:41баллов,
  • Очки рейтинга-8
K
  • KiRi4
  • 7 сентября 2023 г. 17:49

C++ - Тест 001. Первая программа и типы данных

  • Результат:66баллов,
  • Очки рейтинга-1
Последние комментарии
IscanderChe
IscanderChe13 сентября 2023 г. 19:11
Пример использования QScintilla C++ По горячим следам (с другого форума вопрос задали, пришлось в памяти освежить всё) решил дополнить. Качаем исходники с https://riverbankcomputing.com/software/qscintilla/downlo…
Evgenii Legotckoi
Evgenii Legotckoi6 сентября 2023 г. 17:18
Qt/C++ - Урок 048. QThread - работа с потоками с помощью moveToThread Разве могут взаимодействовать объекты из разных нитей как-то, кроме как через сигнал-слоты?" Могут. Выполняя оператор new , Вы выделяете под объект память в куче (heap), …
AC
Andrei Cherniaev5 сентября 2023 г. 13:37
Qt/C++ - Урок 048. QThread - работа с потоками с помощью moveToThread Я поясню свой вопрос. Выше я писал "Почему же в методе MainWindow::on_write_1_clicked() Можно обращаться к методам exampleObject_1? Разве могут взаимодействовать объекты из разных…
n
nvn31 августа 2023 г. 19:47
QML - Урок 004. Сигналы и слоты в Qt QML Здравствуйте! Прекрасный сайт, отличные статьи. Не хватает только готовых проектов для скачивания. Многих комментариев типа appCore != AppCore просто бы не было )))
NSProject
NSProject24 августа 2023 г. 23:40
Django - Урок 023. Like Dislike система с помощью GenericForeignKey Ваша ошибка связана с gettext from django.utils.translation import gettext_lazy as _ Поле должно выглядеть так vote = models.SmallIntegerField(verbose_name=_("Голос"), choices=VOTES) …
Сейчас обсуждают на форуме
IscanderChe
IscanderChe17 сентября 2023 г. 19:24
Интернационализация строк в QMessageBox Странная картина... Сделал минимально работающий пример - всё работает. Попробую на другой операционке. Может, дело в этом.
NSProject
NSProject17 сентября 2023 г. 18:49
Помогите добавить Ajax в проект В принципе ничего сложного с отправкой на сервер нет. Всё что ты хочешь отобразить на странице передаётся в шаблон и рендерится. Ты просто создаёшь файл forms.py в нём описываешь свою форму и в …
BlinCT
BlinCT15 сентября 2023 г. 22:35
Размеры полей в TreeView Всем привет. Пытаюсь сделать дерево вот такого вида Пытаюсь организовать делегат для каждой строки в дереве. ТО есть отступ какого то размера и если при открытии есть под…
IscanderChe
IscanderChe8 сентября 2023 г. 22:07
Кастомная QAbstractListModel и цвет фона, цвет текста и шрифт Похоже надо не абстрактный , а "реальный" типа QSqlTableModel Да, но не совсем. Решилось с помощью стайлшитов и setFont. Спасибо за отлик!
Evgenii Legotckoi
Evgenii Legotckoi6 сентября 2023 г. 16:35
Вопрос: Нужно ли в деструкторе удалять динамически созданные QT-объекты. Напр: Зависит от того, как эти объекты были созданы. Если вы передаёте указатель на parent объект, то не нужно, Ядро Qt само разрулит удаление, если нет, то нужно удалять вручную, иначе будет ут…

Следите за нами в социальных сетях