【问题标题】:How to use the user_passes_test decorator in class based views?如何在基于类的视图中使用 user_passes_test 装饰器?
【发布时间】:2015-06-23 07:56:38
【问题描述】:

我试图在允许用户查看特定用户设置页面之前检查某些条件。我正在尝试使用 user_passes_test 装饰器来实现这一点。该函数位于基于类的视图中,如下所示。我正在使用方法装饰器来装饰视图中的 get_initial 函数。

class UserSettingsView(LoginRequiredMixin, FormView):
    success_url = '.'
    template_name = 'accts/usersettings.html'

    def get_form_class(self):
        if self.request.user.profile.is_student:
            return form1
        if self.request.user.profile.is_teacher:
            return form2
        if self.request.user.profile.is_parent:
            return form3

    @method_decorator(user_passes_test(test_settings, login_url='/accounts/usertype/'))
    def get_initial(self):
        if self.request.user.is_authenticated():
            user_obj = get_user_model().objects.get(email=self.request.user.email)
            if user_obj.profile.is_student:
                return { ..........
       ...... .... 

下面是test_settings函数:

def test_settings(user):
    print "I am in test settings"
    if not (user.profile.is_student or user.profile.is_parent or user.profile.is_teacher):
        return False
    else:
        return True

装饰器出现以下错误。

File "../django/core/handlers/base.py", line 111, in get_response
  response = wrapped_callback(request, *callback_args, **callback_kwargs)
File "../django/views/generic/base.py", line 69, in view
  return self.dispatch(request, *args, **kwargs)
File "../braces/views.py", line 107, in dispatch
  request, *args, **kwargs)
File "../django/views/generic/base.py", line 87, in dispatch
  return handler(request, *args, **kwargs)
File "../django/views/generic/edit.py", line 162, in get
  form = self.get_form(form_class)
File "../django/views/generic/edit.py", line 45, in get_form
  return form_class(**self.get_form_kwargs())
File "../django/views/generic/edit.py", line 52, in get_form_kwargs
  'initial': self.get_initial(),
File "../django/utils/decorators.py", line 29, in _wrapper
  return bound_func(*args, **kwargs)
TypeError: _wrapped_view() takes at least 1 argument (0 given)

我不确定如何解决此错误。我是否将装饰器应用于错误的功能?任何线索都会有所帮助。

【问题讨论】:

    标签: python django python-2.7 decorator python-decorators


    【解决方案1】:

    Django 1.9 具有用于基于类的视图的身份验证混合。您可以按如下方式使用UserPassesTest mixin。

    from django.contrib.auth.mixins import LoginRequiredMixin, UserPassesTestMixin
    
    class UserSettingsView(LoginRequiredMixin, UserPassesTestMixin, View):
        def test_func(self):
            return test_settings(self.request.user)
    
        def get_login_url(self):
            if not self.request.user.is_authenticated():
                return super(UserSettingsView, self).get_login_url()
            else:
                return '/accounts/usertype/'
    

    请注意,在这种情况下,您必须覆盖 get_login_url,因为您希望根据用户未登录或已登录但未通过测试来重定向到不同的 url。

    对于 Django 1.8 及更早版本,您应该装饰 dispatch 方法,而不是 get_initial

    @method_decorator(user_passes_test(test_settings, login_url='/accounts/usertype/')) 
    def dispatch(self, *args, **kwargs):
        return super(UserSettingsView,  self).dispatch(*args, **kwargs)
    

    【讨论】:

    • 对于 Django 2,is_authenticated() 更改为 is_authenticated
    猜你喜欢
    • 1970-01-01
    • 2012-06-11
    • 2014-10-20
    • 2011-08-29
    • 2017-03-08
    • 2013-07-24
    • 2020-07-07
    • 2016-11-09
    相关资源
    最近更新 更多