LoginSignup
7
11

More than 5 years have passed since last update.

[Django]User作成時にUserProfileモデルに空レコードを作成する(Signalの使用)

Last updated at Posted at 2019-04-06

※備忘録

やりたいこと

作成中のアプリにて、サインアップした際にUserモデルにOneToOneで紐付いているUserProfileモデルにレコードを追加する。

環境

  • Python==3.7.0
  • Django==2.1.7

実装

models.py

※Userモデルはemailを使用するためカスタムしています。

models.py
from django.db.models.signals import post_save
from django.dispatch import receiver
from django.contrib.auth.models import PermissionsMixin
from django.contrib.auth.base_user import AbstractBaseUser
from django.contrib.auth.base_user import BaseUserManager

class UserProfile(models.Model):
  """ユーザープロフィールモデル"""
    user = models.OneToOneField(settings.AUTH_USER_MODEL, on_delete=models.CASCADE,
                                related_name='profile')
    picture = models.ImageField(upload_to='profile_pictures', blank=True, null=True,)
    bio = models.TextField(blank=True)

    def __str__(self):
        return self.user.username


class UserManager(BaseUserManager):
    """ユーザーマネージャー."""

    use_in_migrations = True

    def _create_user(self, email, password, **extra_fields):
        """メールアドレスでの登録を必須にする"""
        if not email:
            raise ValueError('メールアドレスを入力してください')
        email = self.normalize_email(email)

        user = self.model(email=email, **extra_fields)
        user.set_password(password)
        user.save(using=self._db)
        return user

    def create_user(self, email, password=None, **extra_fields):
        extra_fields.setdefault('is_staff', False)
        extra_fields.setdefault('is_superuser', False)
        return self._create_user(email, password, **extra_fields)

    def create_superuser(self, email, password, **extra_fields):
        extra_fields.setdefault('is_staff', True)
        extra_fields.setdefault('is_superuser', True)

        if extra_fields.get('is_staff') is not True:
            raise ValueError('Superuser must have is_staff=True.')
        if extra_fields.get('is_superuser') is not True:
            raise ValueError('Superuser must have is_superuser=True.')

        return self._create_user(email, password, **extra_fields)


class User(AbstractBaseUser, PermissionsMixin):
    """カスタムユーザーモデル."""

    email = models.EmailField(_('email address'), unique=True)
    nick_name = models.CharField(_('nick name'), max_length=50, blank=False, unique=True, db_column="ユーザー名")

    is_staff = models.BooleanField(
        _('staff status'),
        default=False,
        help_text=_(
            'Designates whether the user can log into this admin site.'),
    )
    is_active = models.BooleanField(
        _('active'),
        default=True,
        help_text=_(
            'Designates whether this user should be treated as active. '
            'Unselect this instead of deleting accounts.'
        ),
    )
    date_joined = models.DateTimeField(_('date joined'), default=timezone.now)

    objects = UserManager()

    EMAIL_FIELD = 'email'
    USERNAME_FIELD = 'email'
    REQUIRED_FIELDS = []

    class Meta:
        verbose_name = _('user')
        verbose_name_plural = _('users')

    def get_short_name(self):
        return self.nick_name

    @property
    def username(self):
        return self.email


@receiver(post_save, sender=User)
def create_profile(sender, **kwargs):
    """ 新ユーザー作成時に空のprofileも作成する """
    if kwargs['created']:
        user_profile = UserProfile.objects.get_or_create(user=kwargs['instance'])

views.py

views.py
class UserCreate(CreateView):
    """ユーザー登録"""
    model = User
    form_class = UserCreateForm

    def form_valid(self, form):
        """ユーザー登録"""
        # formをテーブルに保存するかを指定するオプション(デフォルト=True)
        user = form.save(commit=True)
        # is_active<-ユーザーアカウントをアクティブにするかどうかを指定,
        # 退会処理も、is_activeをFalseにするという処理がベター。
        user.is_active = True
        user.save()

        return redirect('user_create_done')

実装のポイント

DjangoにはSignalという機能が搭載されていて、なにかアクションが発生したときにそれをフックに特定の処理をキックすることができる。

※該当箇所抜粋

models.py
@receiver(post_save, sender=User)
def create_profile(sender, **kwargs):
    """ 新ユーザー作成時に空のprofileも作成する """
    if kwargs['created']:
        user_profile = UserProfile.objects.get_or_create(user=kwargs['instance'])

@receiverデコレータでpost_saveシグナルを指定し、senderにUserモデルを指定する。

  • post_saveはモデルに対するレコードの追加 or 変更をした直後に動作する。(他にもpre_save、request_startedなどが用意されている)
  • senderで、アクションを検知する対象を指定する。
  • get_or_createメソッドは、引数をキーにDBのレコードを探索し、すでにレコードが存在しない場合、作成し、存在する場合はそのレコードを返す。

create_profile関数では、Userモデルの追加、変更をフックに**kwargsがcreatedであれば、UserProfileモデルにレコードを作成する。

参考にさせていただいた記事

公式ドキュメント
[Django] Signal の使い方まとめ
Djangoシグナル登録の簡単なまとめ
get_or_createの更新時のデフォルト値の扱いなど

7
11
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
7
11