前提・実現したいこと
Djangoにて、emailで認証する形式のコミュニケーションツールを作成しようとしています。
Djangoの開発経験は初めてで、ドキュメントを読みながら進めているレベルです(このようなサービスで質問することも初めてのため、粗相がありましたらご助言いただけると嬉しく思います)。
email認証のカスタムユーザーを作りたく、ドキュメントのチュートリアルとこちらの記事を参考に進めておりましたが、
python manage.py migrateの実行で詰まってしまいました。
stack overflowやteratailでも似たような症状を見かけましたが、そのアンサーを真似しても解決できなかったため、自分で気づいていない問題点や解決策をご教授いただきたいです。
発生している問題・エラーメッセージ
python manage.py migrateの実行で、以下のようなエラーが起きました。
LookupError: App 'floattle' doesn't have a 'User' model. と django.core.exceptions.ImproperlyConfigured: AUTH_USER_MODEL refers to model 'floattle.User' that has not been installed に注目しています。
Traceback (most recent call last): File "/Users/nagamiya/.pyenv/versions/3.6.5/lib/python3.6/site-packages/django/apps/config.py", line 178, in get_model return self.models[model_name.lower()] KeyError: 'user' During handling of the above exception, another exception occurred: Traceback (most recent call last): File "/Users/nagamiya/.pyenv/versions/3.6.5/lib/python3.6/site-packages/django/contrib/auth/__init__.py", line 156, in get_user_model return django_apps.get_model(settings.AUTH_USER_MODEL, require_ready=False) File "/Users/nagamiya/.pyenv/versions/3.6.5/lib/python3.6/site-packages/django/apps/registry.py", line 210, in get_model return app_config.get_model(model_name, require_ready=require_ready) File "/Users/nagamiya/.pyenv/versions/3.6.5/lib/python3.6/site-packages/django/apps/config.py", line 181, in get_model "App '%s' doesn't have a '%s' model." % (self.label, model_name)) LookupError: App 'floattle' doesn't have a 'User' model. During handling of the above exception, another exception occurred: Traceback (most recent call last): File "manage.py", line 21, in <module> main() File "manage.py", line 17, in main execute_from_command_line(sys.argv) File "/Users/nagamiya/.pyenv/versions/3.6.5/lib/python3.6/site-packages/django/core/management/__init__.py", line 401, in execute_from_command_line utility.execute() File "/Users/nagamiya/.pyenv/versions/3.6.5/lib/python3.6/site-packages/django/core/management/__init__.py", line 377, in execute django.setup() File "/Users/nagamiya/.pyenv/versions/3.6.5/lib/python3.6/site-packages/django/__init__.py", line 24, in setup apps.populate(settings.INSTALLED_APPS) File "/Users/nagamiya/.pyenv/versions/3.6.5/lib/python3.6/site-packages/django/apps/registry.py", line 122, in populate app_config.ready() File "/Users/nagamiya/.pyenv/versions/3.6.5/lib/python3.6/site-packages/django/contrib/admin/apps.py", line 24, in ready self.module.autodiscover() File "/Users/nagamiya/.pyenv/versions/3.6.5/lib/python3.6/site-packages/django/contrib/admin/__init__.py", line 26, in autodiscover autodiscover_modules('admin', register_to=site) File "/Users/nagamiya/.pyenv/versions/3.6.5/lib/python3.6/site-packages/django/utils/module_loading.py", line 47, in autodiscover_modules import_module('%s.%s' % (app_config.name, module_to_search)) File "/Users/nagamiya/.pyenv/versions/3.6.5/lib/python3.6/importlib/__init__.py", line 126, in import_module return _bootstrap._gcd_import(name[level:], package, level) File "<frozen importlib._bootstrap>", line 994, in _gcd_import File "<frozen importlib._bootstrap>", line 971, in _find_and_load File "<frozen importlib._bootstrap>", line 955, in _find_and_load_unlocked File "<frozen importlib._bootstrap>", line 665, in _load_unlocked File "<frozen importlib._bootstrap_external>", line 678, in exec_module File "<frozen importlib._bootstrap>", line 219, in _call_with_frames_removed File "/Users/nagamiya/.pyenv/versions/3.6.5/lib/python3.6/site-packages/django/contrib/auth/admin.py", line 6, in <module> from django.contrib.auth.forms import ( File "/Users/nagamiya/.pyenv/versions/3.6.5/lib/python3.6/site-packages/django/contrib/auth/forms.py", line 20, in <module> UserModel = get_user_model() File "/Users/nagamiya/.pyenv/versions/3.6.5/lib/python3.6/site-packages/django/contrib/auth/__init__.py", line 161, in get_user_model "AUTH_USER_MODEL refers to model '%s' that has not been installed" % settings.AUTH_USER_MODEL django.core.exceptions.ImproperlyConfigured: AUTH_USER_MODEL refers to model 'floattle.User' that has not been installed
ファイルの構造
floattleがアプリ名称です。スクリーンショットで失礼いたします。
該当のソースコード
settings.pyの抜粋です。AUTH_USER_MODEL〜の後にはMIDDLEWAREやROOT_URLCONF、TEMPLATESなどが記述してあります。
Python
1import os 2 3BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) 4 5SECRET_KEY = 'qg27x8s5ju-61*6jdp$$@s3@9=7rv63^f0x9uvl+ukl^fkt+_e' 6 7DEBUG = True 8 9ALLOWED_HOSTS = [] 10 11# Application definition 12 13INSTALLED_APPS = [ 14 'floattle.apps.FloattleConfig', 15 'django.contrib.admin', 16 'django.contrib.auth', 17 'django.contrib.contenttypes', 18 'django.contrib.sessions', 19 'django.contrib.messages', 20 'django.contrib.staticfiles', 21] 22 23# カスタムユーザクラスを使用 24AUTH_USER_MODEL = 'floattle.User'
models.pyです。
Python
1from django.db import models 2from django.core.mail import send_mail 3from django.contrib.auth.validators import UnicodeUsernameValidator 4from django.utils import timezone 5from django.contrib.auth.models import PermissionsMixin, UserManager 6from django.contrib.auth.base_user import AbstractBaseUser 7from django.utils.translation import gettext_lazy as _ 8 9 10# usernameではなくemailで認証をする 11class UserManager(UserManager): 12 use_in_migrations = True 13 14 def _create_user(self, email, password, **extra_fields): 15 """ 16 Create and save a user with the given username, email, and password. 17 """ 18 if not email: 19 raise ValueError('The given email must be set') 20 email = self.normalize_email(email) 21 user = self.model(email=email, **extra_fields) 22 user.set_password(password) 23 user.save(using=self._db) 24 return user 25 26 def create_user(self, email=None, password=None, **extra_fields): 27 extra_fields.setdefault('is_staff', False) 28 extra_fields.setdefault('is_superuser', False) 29 return self._create_user(email, password, **extra_fields) 30 31 def create_superuser(self, email, password, **extra_fields): 32 extra_fields.setdefault('is_staff', True) 33 extra_fields.setdefault('is_superuser', True) 34 35 if extra_fields.get('is_staff') is not True: 36 raise ValueError('Superuser must have is_staff=True.') 37 if extra_fields.get('is_superuser') is not True: 38 raise ValueError('Superuser must have is_superuser=True.') 39 40 return self._create_user(email, password, **extra_fields) 41 42 43class User(AbstractBaseUser, PermissionsMixin): 44 name_validator = UnicodeUsernameValidator() 45 name = models.CharField( 46 _('name'), 47 max_length=30, 48 help_text=_('Required. 30 characters or fewer. Letters, digits and @/./+/-/_ only.' 49 ), 50 validators=[name_validator] 51 ) 52 email = models.EmailField(_('email address'), unique=True) 53 is_staff = models.BooleanField( 54 _('staff status'), 55 default=False, 56 help_text=_( 57 'Designates whether the user can log into this admin site.'), 58 ) 59 is_active = models.BooleanField( 60 _('active'), 61 default=True, 62 help_text=_( 63 'Designates whether this user should be treated as active. ' 64 'Unselect this instead of deleting accounts.'), 65 ) 66 date_joined = models.DateTimeField(_('date joined'), default=timezone.now) 67 last_login = models.DateTimeField( 68 _('date last login'), default=timezone.now()) 69 70 objects = UserManager() 71 72 EMAIL_FIELD = 'email' 73 USERNAME_FIELD = 'email' 74 REQUIRED_FIELDS = [] 75 76 class Meta: 77 verbose_name = _('user') 78 verbose_name_plural = _('users') 79 abstract = True 80 def clean(self): 81 super().clean() 82 self.email = self.__class__.objects.normalize_email(self.email) 83 84 def email_user(self, subject, message, from_email=None, **kwargs): 85 ''' Send an email to this user. ''' 86 send_mail(subject, message, from_email, [self.email], **kwargs) 87 88 @property 89 def username(self): 90 ''' username属性のgetter ''' 91 return self.email 92 93 94class Post(models.Model): 95 user = models.ForeignKey('User', on_delete=models.CASCADE) 96 text = models.TextField(max_length=400) 97 shed_count = models.IntegerField() 98 is_active = models.BooleanField(default=True) 99 keep = models.ManyToManyField(User) 100 date_published = models.DateTimeField(blank=True, null=True) 101 date_deleted = models.DateTimeField(blank=True, null=True) 102 103 ''' 文字数によって紙のデザインが変わるようにそのうちしたいので定義しておいた ''' 104 def length(self): 105 return len(self.text) 106
試したこと
- settings.pyの編集
INSTALLED_APPSに追記する部分を変えてみました。
Python
1INSTALLED_APPS = [ 2 'django.contrib.admin', 3 'django.contrib.auth', 4 'django.contrib.contenttypes', 5 'django.contrib.sessions', 6 'django.contrib.messages', 7 'django.contrib.staticfiles', 8 'floattle', #ここです 9]
- apps.pyの確認
Python
1from django.apps import AppConfig 2 3class FloattleConfig(AppConfig): 4 name = 'floattle'
- admin.pyの編集
上記の参考サイトを真似しつつ、必要な部分を変えて書いていました。このファイルが原因...?と思い、書いた状態とデフォルトの状態で試してみましたが変化がありませんでした。
- マイグレーションとDBの削除
マイグレーションが実行すらできていない状態のため不要かと思いましたが、念のために試してみました。
- プロジェクトごと作り直し
補足情報(FW/ツールのバージョンなど)
Django 3.0.8
Python 3.6.5
MySQL
仮想環境は今回使っておりません...(これは何か関係しているのでしょうか)
あなたの回答
tips
プレビュー