swor
swor

Reputation: 901

ImportError: cannot import name 'User'

I created custom save method for my class User. Because I have to create DriverRegistration object when I create User object.

When I added only method save i received an error the code works fine without this method

 ImportError: cannot import name 'User' from 'backend.apps.users.models' (/app/backend/apps/users/models/__init__.py)

How can I fix the error?

Init file looks fine:

from .user import User
from .driver_registration import DriverRegistration

user.py

    class User(AbstractBaseUser, PermissionsMixin, TimeStampMixin):
        objects = UserManager()
    
        first_name = models.CharField(
            "first name", max_length=30, blank=True, null=True
        )
        ...
    
        def save(self, *args, **kwargs):
            super().save(*args, **kwargs)  # Create User object
            driver_registration = DriverRegistration.objects.create(user=self.pk)
            driver_registration.save()


    class UserManager(BaseUserManager):
        def create_user(
            self,
            phone_number,
            password,
            device_id=None,
            first_name=None,
            last_name=None,
        ):
            if not phone_number:
                raise ValueError("Users must have an phone number")
    
            user = self.model(
                phone_number=phone_number,
                device_id=device_id,
                first_name=first_name,
                last_name=last_name,
            )
            user.is_staff = True
            user.set_password(password)
            user.save(using=self._db)
            return user
    
        def create_superuser(self, phone_number, password):
            user = self.create_user(phone_number, password=password)
            user.is_superuser = True
            user.user_type = UserType.USER_TYPE_MANAGER
            user.device_id = hashlib.md5(
                str(random.getrandbits(32)).encode("utf-8")
            ).hexdigest()
            user.save(using=self._db)
            return user
    

driver_registration.py

class DriverRegistration(models.Model):
    user = models.ForeignKey(
        User,
        on_delete=models.CASCADE,
        null=True,
    )

Full error back trace

File "/app/backend/apps/users/models/__init__.py", line 1, in <module>
^[[36mshashka-backend    |^[[0m     from .user import User
^[[36mshashka-backend    |^[[0m   File "/app/backend/apps/users/models/user.py", line 19, in <module>
^[[36mshashka-backend    |^[[0m     from backend.apps.users.models.driver_registration import DriverRegistration
^[[36mshashka-backend    |^[[0m   File "/app/backend/apps/users/models/driver_registration.py", line 2, in <module>
^[[36mshashka-backend    |^[[0m     from backend.apps.users.models import User
^[[36mshashka-backend    |^[[0m ImportError: cannot import name 'User' from 'backend.apps.users.models' (/app/backend/apps/users/models/__init__.py)

Upvotes: 0

Views: 967

Answers (1)

Nicky
Nicky

Reputation: 316

Just as a matter of implementation - I think you would be better using Django signals compared to overriding the save method.

Here's an example:

@receiver(post_save, sender=User)
def save_driver_registration(sender, instance, created, **kwargs):
    if created:
        DriverRegistration.objects.create(user=instance.pk)

This will create a new DriverRegistration object when a new User entry has been created. This is arguably a better approach than overriding .save() because with your implementation, every time you .save() a user (for example - updating a user's name/password) - you will create a new DriverRegistration object.

Read more here: https://docs.djangoproject.com/en/3.1/topics/signals/

Upvotes: 1

Related Questions