0

最近、Django アプリでクラスベースのビューに切り替えましたが、それらをできるだけエレガントに使用したいと考えています。アプリにはコメント システムがあり、権限が一致する場合、管理者はコメントを削除および/または公開/非公開にすることができます。RedirectView私は両方の機能を含めて、そのために書いた:

class CommentChangeView(RedirectView, SingleObjectMixin):
    """
    A redirection that acts on a Comment. The url parameter
    "action" is taken as a class function and executed.
    """

    model = Comment


    def get_redirect_url(self, pk):
        """
        Redirect to the article page, always.
        """
        return reverse('post', args=(self.object.post.slug,))


    def get(self, *args, **kwargs):
        """
        Here, it is decided what to execute.
        """

        self.object = self.get_object()
        func = getattr(self, kwargs.pop('action', None), None)

        if callable(func):
            func()

        return super(CommentChangeView, self).get(*args, **kwargs)

    @method_decorator(permission_required('blog.delete_comment'))
    def delete(self):
        """
        Delete the comment
        """

        self.object.delete()
        messages.success(self.request, 'Comment deleted.')

    @method_decorator(permission_required('blog.change_comment'))
    def toggle_publish(self):
        """
        Toggle its publication state
        """

        self.object.published = not self.object.published
        self.object.save()
        messages.success(self.request, 'Comment toggled.')

問題は、2 つのアクションに異なる権限を持たせたいということです。つまり、異なるデコレータです。通常、dispatch関数は装飾されています。上記のコードは機能しませんTypeError。. デコレータがなくても、完全に機能します。

このケースをどのように実装しますか?または、削除と公開のビューを分離する必要がありますか?


の StackTrace は次のTypeErrorとおりです。

Internal Server Error: /comment/toggle_publish/1/
Traceback (most recent call last):
  File "/***/lib/python2.7/site-packages/django/core/handlers/base.py", line 115, in get_response
    response = callback(request, *callback_args, **callback_kwargs)
  File "/***/lib/python2.7/site-packages/django/views/generic/base.py", line 68, in view
    return self.dispatch(request, *args, **kwargs)
  File "/***/lib/python2.7/site-packages/django/views/generic/base.py", line 86, in dispatch
    return handler(request, *args, **kwargs)
  File "/***/blog/blog/views.py", line 186, in get
    func()
  File "/***/lib/python2.7/site-packages/django/utils/decorators.py", line 25, in _wrapper
    return bound_func(*args, **kwargs)
TypeError: _wrapped_view() takes at least 1 argument (0 given)
4

1 に答える 1

0

Bibhas は、StackTrace を詳しく調べて、私を正しい方向に導きました。関数に追加の引数を 1 つ導入することで、機能させることができました。このコードは現在動作します:

class CommentChangeView(RedirectView, SingleObjectMixin):
    """
    A redirection that acts on a Comment. The url parameter
    "action" is taken as a class function and executed. It therefore
    combines the delete and publish_comment functions (and, later, possibly)
    more.
    """

    model = Comment


    def get_redirect_url(self, pk):
        """
        Redirect to the article page, always.
        """
        return reverse('post', args=(self.object.post.slug,))


    def get(self, request, *args, **kwargs):
        """
        Here, it is decided what to execute.
        """

        self.object = self.get_object()
        func = getattr(self, kwargs.pop('action', None), None)

        if callable(func):
            func(request)

        return super(CommentChangeView, self).get(request, *args, **kwargs)

    @method_decorator(permission_required('blog.delete_comment'))
    def delete(self, request):
        """
        Delete the comment
        """

        self.object.delete()
        messages.success(self.request, 'Comment deleted.')

    @method_decorator(permission_required('blog.change_comment'))
    def toggle_publish(self, request):
        """
        Toggle its publication state
        """

        self.object.published = not self.object.published
        self.object.save()
        messages.success(self.request, 'Comment toggled.')
于 2013-04-06T09:44:10.763 に答える