Как правильно выполнить запрос по многим многим модельным полям? - PullRequest
1 голос
/ 15 апреля 2020

Хорошо, вот что я пытаюсь сделать, как разработчик noob ie. Я пытаюсь создать почтовый ящик для пользователя. У меня есть модель Conversation, которая имеет поле «многие ко многим», а затем у меня есть модель InstantMessanger, в которой есть поле sender_id, date и receive_id, которое является внешним ключом для Conversation. Я пытаюсь сначала перебрать Conversation и отобразить членов, с которыми у пользователя был convo's, а затем отобразить ссылку, по которой вы можете перейти к фактическому содержанию этого сообщения. Тем не менее, я застрял в том, чтобы заставить участников даже показать. Я попробовал несколько вещей здесь и огляделся вокруг, но я продолжаю получать ошибки, такие как модель «разговора» не повторяемая или не отображаемая, а отображаемая как «Dating_app.Profile.none», объект и так далее. Так что я застрял здесь и не уверен, как это сделать. Какой правильный способ сделать это?

views.py

def messages(request,profile_id):

    messages = InstantMessage.objects.filter(Q(sender_id=request.user)).\
    values('sender_id','receiver_id', 'message', 'date', ).\
    order_by('date',)


    profile = get_object_or_404(Profile,id=profile_id)

    conversations = Conversation.objects.filter(members=request.user)


return render(request, 'dating_app/messages.html', {'messages': messages,'profile': profile,'conversations':conversations,})

сообщений. html

{% for conversation in conversations %}
    <li class="text-right list-group-item">
        {% for members in conversation %}{% if members != request.user %}
            {{ members.username }}<br/>
        {% endif %}{% endfor %}

    </li>
{%endfor %}

models.py

class ProfileManager(BaseUserManager):



    def create_user(self, username, email,description,photo, password=None):
        if not email:
            raise ValueError("You must creat an email")
        if not username:
            raise ValueError("You must create a username!")
        if not description:
            raise ValueError("You must write a description")
        if not photo:
            raise ValueError("You must upload a photo")

        user = self.model(
                email=self.normalize_email(email),
                username = username, 
                description= description,
                photo= photo,

            )

        user.set_password(password)
        user.save(using=self._db)
        return user 


    def create_superuser(self, username, email,description,photo, password):
        user = self.create_user(
                email=self.normalize_email(email),
                password=password,
                username=username,
                description=description,
                photo=photo,

            )

        user.is_admin=True
        user.is_staff=True
        user.is_superuser=True
        user.save(using=self._db)
        return user




class Profile(AbstractBaseUser):

    class Meta:
        swappable = 'AUTH_USER_MODEL'


    email                       = models.EmailField(verbose_name="email")
    username                    = models.CharField(max_length=30, unique=True)
    date_joined                 = models.DateTimeField(verbose_name='date joined', auto_now_add=True)
    last_login                  = models.DateTimeField(verbose_name='last login', auto_now=True)
    is_admin                    = models.BooleanField(default=False)
    is_active                   = models.BooleanField(default=True)
    is_staff                    = models.BooleanField(default=False)
    is_superuser                = models.BooleanField(default=False)
    #what I added
    description                 = models.TextField()
    photo                       = models.ImageField(upload_to='profile_photo',blank=False, height_field=None, width_field=None, max_length=100)
    matches                     = models.ManyToManyField(settings.AUTH_USER_MODEL, related_name='+', blank=True)



    USERNAME_FIELD = 'username'
    REQUIRED_FIELDS = ['description','photo','email']


    objects = ProfileManager()


    def __str__(self):
        return self.username


    def has_perm(self, perm, obj=None):
        return self.is_admin


    def has_module_perms(self,app_label):
        return True



class UserVote(models.Model):

    user = models.ForeignKey(settings.AUTH_USER_MODEL, on_delete=models.CASCADE)
    voter = models.ForeignKey(settings.AUTH_USER_MODEL, related_name='given_vote', on_delete=models.CASCADE)
    vote = models.BooleanField(default=False)

    class Meta:
        unique_together = (('user', 'voter'))


class Conversation(models.Model):
    members = models.ManyToManyField(settings.AUTH_USER_MODEL)




class InstantMessage(models.Model):

    sender = models.ForeignKey(settings.AUTH_USER_MODEL, related_name= 'senderr',on_delete=models.CASCADE )
    receiver = models.ForeignKey(Conversation, on_delete=models.CASCADE)
    message = models.TextField()
    date = models.DateTimeField(auto_now_add=True)


    def __unicode__(self):
        return self.message

1 Ответ

1 голос
/ 15 апреля 2020

Вы можете отобразить участников с помощью:

{% for conversation in conversations %}
    <li class="text-right list-group-item">

    {% for member in conversation<b>.members.all</b> %}
        {% if member != request.user %}
            {{ member.username }}<br/>
        {% endif %}
    {% endfor %}
    <li>
{% endfor %}

Это, однако, приведет к проблеме N + 1 , когда для показа членов вы сделаете дополнительный запрос за разговор , что не очень эффективно. Вы можете повысить эффективность с помощью .prefetch_related(..) [Django -doc] :

def messages(request,profile_id):
    messages = InstantMessage.objects.filter(Q(sender_id=request.user)).\
    values('sender_id','receiver_id', 'message', 'date', ).\
    order_by('date',)
    profile = get_object_or_404(Profile,id=profile_id)

    conversations = Conversation.objects.filter(
        members=request.user
    )<b>.prefetch_related('members')</b>

    return render(request, 'dating_app/messages.html', {'messages': messages,'profile': profile,'conversations':conversations,})

РЕДАКТИРОВАТЬ : вы можете заказать сообщения по последнему полученному сообщению (в порядке убывания) с:

from django.db.models import Max

def messages(request,profile_id):
    messages = InstantMessage.objects.filter(Q(sender_id=request.user)).\
    values('sender_id','receiver_id', 'message', 'date', ).\
    order_by('date',)
    profile = get_object_or_404(Profile,id=profile_id)

    conversations = Conversation.objects.filter(
        members=request.user
    ).annotate(
        last_message=Max('instantmessage__date')
    ).prefetch_related('members')<b>.order_by(</b>
        '-last_message'
    <b>)</b>

    return render(request, 'dating_app/messages.html', {'messages': messages,'profile': profile,'conversations':conversations,})
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...