Search code examples
pythondjangocomments

Comment model only for user django (1048, “Column 'user_id' cannot be null” or no user filled)


I hope you are well.

I'm trying to create comment for user with one form field (content). I'd like to have the user field automatically filled in user value.

I. I started with this model but I got an error (1048, “Column 'user_id' cannot be null”): models.py

class Comment(models.Model):
    post = models.ForeignKey(Post,on_delete=models.CASCADE,related_name="comments")
    user = models.ForeignKey(User,on_delete=models.CASCADE)
    content = models.TextField(max_length=160)
    publishing_date = models.DateField(auto_now_add=True)
     
    def __str__(self):
        return self.post.title

views.py

class PostDetail(generic.DetailView,FormMixin):
    model = Post
    context_object_name = 'post'
    template_name = 'post_detail.html'
    form_class = CreateCommentForm
    
    def get_context_data(self, **kwargs):
        context = super(PostDetail, self).get_context_data(**kwargs)
        context['form'] = self.get_form()
        return context

    def form_valid(self, form):
        if form.is_valid():
            form.instance.post = self.object
            form.save()
            return super(PostDetail, self).form_valid(form)
        else:
            return super(PostDetail, self).form_invalid(form)

    def post(self,*args,**kwargs):
        self.object = self.get_object()
        form = self.get_form()
        if form.is_valid():
            return self.form_valid(form)
        else:
            return self.form_valid(form)

    def get_success_url(self):
        return reverse('post_detail',kwargs={"slug":self.object.slug})

forms.py

class CreateCommentForm(forms.ModelForm):
    def __init__(self,*args,**kwargs):
        super(CreateCommentForm, self).__init__(*args,**kwargs)
        self.helper = FormHelper()
        self.helper.form_method="post"
        self.helper.layout = Layout(
            Field("content",css_class="form-control",style="margin-bottom:10px",rows="1"),
        )
        
        self.helper.add_input(Submit('submit','Comment',css_class="btn btn-sm",style="background-color: #0d6ec5;border-color: #0d6ec5;"))



    class Meta:
        model = Comment
        fields = [
            'content'
        ]

II. It worked from my admin panel but not from my website (I got this error: (1048, “Column 'user_id' cannot be null”)). So I've decided to change my models with:

class Comment(models.Model):
    post = models.ForeignKey(Post,on_delete=models.CASCADE,related_name="comments")
    user = models.ForeignKey(User, blank=True, null=True, on_delete=models.CASCADE)
    content = models.TextField(max_length=160)
    publishing_date = models.DateField(auto_now_add=True)
     
    def __str__(self):
        return self.post.title

The thing is if a user post a comment, content is ok but there is no user related to the comment. Anyone has an idea?


Solution

  • You need to specify the .user attribute, probably to the logged in user:

    from django.contrib.auth.mixins import LoginRequiredMixin
    
    class PostDetail(LoginRequiredMixin, FormMixin, generic.DetailView):
        model = Post
        context_object_name = 'post'
        template_name = 'post_detail.html'
        form_class = CreateCommentForm
        
        def get_context_data(self, **kwargs):
            context = super().get_context_data(**kwargs)
            context['form'] = self.get_form()
            return context
    
        def form_valid(self, form):
            form.instance.post = self.object
            form.instance.user = self.request.user
            return super().form_valid(form)
    
        def post(self,*args,**kwargs):
            self.object = self.get_object()
            form = self.get_form()
            if form.is_valid():
                return self.form_valid(form)
            else:
                return self.form_valid(form)
    
        def get_success_url(self):
            return reverse('post_detail',kwargs={"slug":self.object.slug})

    Note: You can limit views to a class-based view to authenticated users with the LoginRequiredMixin mixin [Django-doc].


    Note: It is normally better to make use of the settings.AUTH_USER_MODEL [Django-doc] to refer to the user model, than to use the User model [Django-doc] directly. For more information you can see the referencing the User model section of the documentation.