概要
- Djangoで、複数のユーザ種別を作れるシステムが欲しい
- 「サプライユーザ」「バイヤーユーザ」の2種類のユーザを作れるようにする
- ユーザ種別ごとに、異なる属性を持たせる
- サプライユーザは会社名、バイヤーユーザは最寄り駅を登録できるようにする
- 環境は、
- Python 3.6.9
- Django 3.1.1
- django-allauth 0.42.0 (ソーシャルログイン機能)
Python + Django の制限
- ユーザ認証(ログインやサインアップ)に使用できるモデルクラスは1つだけ
- settings.pyの
AUTH_USER_MODEL
で指定する
- settings.pyの
実装の概要
- CustomUserモデルを1つ作り、userTypeをもたせる
- ユーザ種別ごとの情報は別テーブルに持ち、カスタムユーザクラスと
OneToOneField
で紐付ける- UserDetailSupplier
- UserDetailBuyer
- ユーザの保存にはアダプタを使う。
- AccountAdapterを
allauth.account.adapter.DefaultAccountAdapter
を継承して作成する -
save_user
メソッドを実装し、その中でuserTypeごとに保存処理を分ける - settings.py の
ACCOUNT_ADAPTER
で上記クラスを指定する
- AccountAdapterを
- サインアップ用のテンプレートは種別ごとに分け
るが、送信先は1つ、送信先も2つに分ける。-
signup
とsignup_supplier
を作成
-
- ログイン用のテンプレートは1つ
django-allauth と自作クラスの関係
- django-allauthは
account
というアプリ名を使っているが、それとは別にmember
というアプリを作り、そこで各種管理を行う- ググると、複数形の
accounts
という名前で作る例が多いが、allauth側と区別がつけづらいのと、ここだけ複数形にしたくなかったので、別な名前をつけた- Pythonのメンバ関数と似ているが、アプリ名だから混同しないだろうという考え
- ググると、複数形の
問題点
-
サインアップ時にアカウント重複などでエラーとなった場合に、追加したテンプレートではなく元からあるテンプレートに戻ってしまう-
signup_supplier
でサインアップしようとして失敗すると、signup
に遷移してしまう- フォームの送信先を2つに分けて解決。
-
- 追加した項目をフォームから取得する際に
request.POST
を使っているが、どうにかならないか-
form.cleaned_data.get
から取得しようとしたが、空で取得できなかった
-
- ユーザ情報のテーブルが分かれるので、admin画面が分かりづらい。管理画面を自作したほうがよさそう
実装の具体例
django-allauthのインストール
- pipでインストールし、各種設定を行う
設定
- 設定ファイルに以下を追加する
# settings.py
# 認証に使うモデルを指定
AUTH_USER_MODEL = 'member.CustomUser'
# signupformからの情報をcustomusermodelに保存するためのアダプタを指定
ACCOUNT_ADAPTER = 'member.adapter.AccountAdapter'
モデルクラスの作成
# member/models.py
from django.contrib.auth.models import AbstractUser
from django.db import models
from django.contrib.auth.models import PermissionsMixin, UserManager
class UserType(models.Model):
""" ユーザ種別 """
typename = models.CharField(verbose_name='ユーザ種別',
max_length=150)
def __str__(self):
return f'{self.id} - {self.typename}'
USERTYPE_SUPPLIER = 100
USERTYPE_BUYER = 200
USERTYPE_DEFAULT = USERTYPE_BUYER
class CustomUserManager(UserManager):
""" 拡張ユーザーモデル向けのマネージャー """
def _create_user(self, email, password, **extra_fields):
if not email:
raise ValueError('The given email must be set')
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 CustomUser(AbstractUser):
""" 拡張ユーザーモデル """
class Meta(object):
db_table = 'custom_user'
#作成したマネージャークラスを使用
objects = CustomUserManager()
# モデル内にユーザ種別を持つ
userType = models.ForeignKey(UserType,
verbose_name='ユーザ種別',
null=True,
blank=True,
on_delete=models.PROTECT)
def __str__(self):
return self.username
class UserDetailSupplier(models.Model):
user = models.OneToOneField(CustomUser,
unique=True,
db_index=True,
related_name='detail_supplier',
on_delete=models.CASCADE)
# サプライヤーユーザ向けの項目
companyName = models.CharField(
max_length=100,
null=True,
blank=True,
)
def __str__(self):
user = CustomUser.objects.get(pk=self.user_id)
return f'{user.id} - {user.username} - {user.email} - {self.id} - {self.companyName}'
class UserDetailBuyer(models.Model):
user = models.OneToOneField(CustomUser,
unique=True,
db_index=True,
related_name='detail_buyer',
on_delete=models.CASCADE)
# バイヤーユーザ向けの項目
nearestStation = models.CharField(
max_length=100,
null=True,
blank=True,
)
def __str__(self):
user = CustomUser.objects.get(pk=self.user_id)
return f'{user.id} - {user.username} - {user.email} - {self.id} - {self.nearestStation}'
アダプタの作成
# member/adapter.py
from allauth.account.adapter import DefaultAccountAdapter
from .models import *
class AccountAdapter(DefaultAccountAdapter):
def save_user(self, request, user, form, commit=True):
"""
This is called when saving user via allauth registration.
We override this to set additional data on user object.
"""
# Do not persist the user yet so we pass commit=False
# (last argument)
user = super(AccountAdapter, self).save_user(request, user, form, commit=False)
#user.userType = form.cleaned_data.get('userType')
user.userType = UserType(request.POST['userType'])
if not user.userType:
user.userType = UserType(USERTYPE_DEFAULT) # デフォルトのユーザ種別を設定
# ユーザIDを取得するために一旦保存する
user.save()
if int(user.userType.id) == USERTYPE_SUPPLIER:
# サプライヤーユーザ
supplier = UserDetailSupplier()
supplier.user_id = user.id
supplier.companyName = request.POST['companyName']
supplier.save()
else:
# それ以外は一般ユーザ
user.userType = UserType(USERTYPE_BUYER)
buyer = UserDetailBuyer()
buyer.user_id = user.id
buyer.nearestStation = request.POST.get('nearestStation', False)
buyer.save()
テンプレートの作成
- django-allauthのテンプレートを上書き
- templates/account/signup.html
- templates/account/signup_supplier.html
- formにhiddenでユーザ種別を入れる。
<input type="hidden" name="userType" value="1" />
- POST先は、どちらも
{% url 'account_signup' %}
とする
- formにhiddenでユーザ種別を入れる。
urls.pyの設定
- 以下を追加
path('member/signup_supplier/', TemplateView.as_view(template_name = 'account/signup_supplier.html'), name='signup_supplier'),
その他
- admin.pyに、各モデルクラスを追加しておく
参考
-
googleで「django-allauth userType」で検索
https://www.google.com/search?q=django-allauth+userType&oq=django-allauth+userType&aqs=chrome..69i57.9683j0j1&sourceid=chrome&ie=UTF-8