程序少不了验证用户与权限分配。通过 Django 自带以及我们一些扩展就能够满足验证与权限的需求。

我在使用 Django 遇到的“login(request, user) 之后,再重定向这个 request,然后 request.user 变成了 anonymous user”问题。

使用了一个自定义的 backend,然后使用 authenticate(parm) 能够获取到 user。

搜索了这些

django login in and redirect to current_url - Google 搜索

django 登陆之后丢失信息 - Google 搜索
Redirect to the originally requested page after django registration and email activation | Mitch Fournier
Django:Redirect after login success - Stack Overflow
django redirect after login in without user - Google 搜索
django redirect after login not working - Google 搜索

django last login session - Google 搜索

找了几小时都没有找到答案,一直尝试 login(request, user),每次 login 成功了,回来又是 request user。我大部分时间怀疑 login() 方法是不是有问题,但并没有什么收获。然后看了看自定义的 backend,没发现问题,并且通过 authenticate 能够拿到 user。然后问了同事,他们也找到问题。

问了别人都没解决,那只能再次自己动手了。这次我想从底层弄起,想 http 是无状态的,只能通过 cookie, session 来获取状态。那么Django登录一定是在 session 中做文章。于是把 Django 的session变为 dict 查看,发现了问题

session 中存在 _auth_user_id,但是 request.user=AnonymousUser: AnonymousUser

dict_items([('_auth_user_backend', 'apps.traffic.backends.WeixinAccessTokenAuthenticationBackend'), ('openid', 'ouRGywq5gA9bTyMvAw-JcovC6M00'), ('_auth_user_id', '254'), ('_auth_user_hash', '2f27f6ec2895b333caa320790f7868d87094017c'), ('code', '0412PZ1T1Uj85a1HGK1T1BvY1T12PZ1o'), ('_session_expiry', 2592000)])

于是又问同事,他说是 Django 中间件那里过不了。我看了下 backend,能通过 authenticate 获取 user 啊。没办法,再次 Google "django session and login in",这里http://eli.thegreenplace.net/2011/07/17/django-sessions-part-iii-user-authentication 说道

Auth middleware
Similarly to the path we've taken with sessions, it's instrumental to first see how authentication middleware is implemented. Or in other words, how did the "user" attribute get into my HTTP request?

The answer is in contrib/auth/middleware.py [2]:

  1. class LazyUser(object):
  2. def __get__(self, request, obj_type=None):
  3.   if not hasattr(request, '_cached_user'):
  4.     from django.contrib.auth import get_user
  5.     request._cached_user = get_user(request)
  6.   return request._cached_user

于是搜索显示调用 `get_user(request)`发现返回的是 None,通过 Pycharm 打断点进去看,确认问题出在自定义 backend 上面,其中的 get_user 方法没有返回 user 对象,返回的是 None。问题终结!

反思一下为什么解决这个问题花了这么长时间:

没有规矩。

找 bug 随便找,并没有指定很好的方案。一直在 login 那里打转,不相信是其他的问题。

如果再让我来一次,我会调整为:

按照靠谱程度找 bug。自定义的代码比 Django 源码更不靠谱,所以先重点关注自定义的代码。

从底层入手。authenticate 的实现机制是什么,然后一层层往上。

认真的看代码,可以从下往上看,提高关注度,而不是扫一遍,觉得大概是没错的。

看一遍Django自定义 backend 是怎么写的,并与已有代码比较。

找完了 bug,进入正文。

Django 的认证方式:

middleware, 自定义基类, 装饰器以及Mixin

middleware

middleware 是笼罩所有视图的,比如我这有一个充值话费的项目,就使用了 middleware 来进行验证

自定义基类

自定义基类,我之前用过,感觉不灵活,是我自己想出来的,与 Django 的体系应该不同

装饰器以及Mixin

这个也属于 Django 的体系,也是我们常用的。有些页面需要登录就加装饰器,有些不需要就不加。

参考:

How Django sessions work - introduction  (推荐看完这个系列)

django authenticate的更多相关文章

  1. python - django authenticate 返回结果一直是None

    # 不论怎么测试 authenticate 返回结果一直是None,查了查原因好像是 django 2.1 版本的问题 ( 我用的是 2.1.7) 解决方法: 1. 先导入 from django.c ...

  2. Django学习系列之重写User模型和登录验证

    重写User模型 Django内置的User模型可能不适合某些项目,我们可能要基于内置的添加一些字段 创建users app startapp users 修改settings.py配置文件,覆盖默认 ...

  3. 『Django』第N+1节: Django自带的认证系统 - auth

    个人网站: lipeiguan.top 以后会慢慢转移到个人网站, 欢迎大家收藏^ . ^ 写在前面 我们在开发一个网站的时候, 经常需要实现网站的用户系统. 这个时候我们需要实现用户注册.用户登录. ...

  4. django 的auth.authenticate返回为None

    使用auth.authenticate(username= username,passowrd=passowrd),这个用户认证时候,明明数据库中有记录,但是返回就None 我的错误点比较多: 1.我 ...

  5. Django 自带登录验证:authenticate和login,login_require,logout模块

    验证之前需要在settings 中指定验证数据models AUTH_USER_MODEL = 'crm.UserProfile'#app名字.表名字 1.authenticate是一个方法,验证账号 ...

  6. Django中authenticate和login模块

    Django 提供内置的视图(view)函数用于处理登录和退出,Django提供两个函数来执行django.contrib.auth中的动作 : authenticate()和login(). 认证给 ...

  7. django 自带认证系统(login,logout,authenticate,login_required)

    from django.contrib.auth import login,authenticate,logoutfrom django.contrib.auth.decorators import ...

  8. 《Django By Example》第四章 中文 翻译 (个人学习,渣翻)

    书籍出处:https://www.packtpub.com/web-development/django-example 原作者:Antonio Melé (译者注:祝大家新年快乐,这次带来<D ...

  9. Django

    一.Django 简介 Django 是一个由 Python 写成的开放源代码的 Web 应用框架.它最初是被开发来用于管理劳伦斯出版集团旗下的一些以新闻内容为主的网站的,即是 CMS(内容管理系统) ...

随机推荐

  1. Docker化运维方式讲解

    应用迁移需求 应用运维需要考虑的一个重要问题就是迁移, 在不同机器.机房.环境间迁移.迁移的原因有很多, 比如硬件过保(硬件故障), 机房迁移, 应用扩缩容等. 应用迁移的核心需求是: 简单.迁移操作 ...

  2. ASP.NET MVC 简介

    1. ASP.NET MVC 是什么? ASP.NET MVC是微软官方提供的以MVC模式为基础的ASP.NET Web应用程序(Web Application)框架,它由Castle的MonoRai ...

  3. 1-2 nodejs小节 文件读取

    1.表达式 在命令行输入  node回车后,可以在后边输入相应的表达式,进行运算操作   2.阻塞文件读取 var data=fs.readFileSync('input.txt', 'utf-8') ...

  4. 写pdf文件

    using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; usin ...

  5. 详细解读DialogFragment

    原博客地址:http://www.cnblogs.com/tianzhijiexian/p/4161811.html 相信看这篇文章的人都应该知道android中的Dialog了吧,如果对于Dialo ...

  6. SQL SERVER 2000 迁移后SQL SERVER代理服务启动错误分析

    公司有一个老系统,这个系统所用的数据库是SQL SERVER 2000,它所在的Dell服务器已经运行超过10年了,早已经过了保修服务期,最近几乎每周会出现一次故障,加之5月份另外一台服务器坏了两个硬 ...

  7. Failed to retrieve data for this request. (Microsoft.SqlServer.Management.Sdk.Sfc)

    使用Microsoft SQL SERVER 2014 Management Studio访问Azure SQL Database时,查看存储过程时遇到下面错误信息: TITLE: Microsoft ...

  8. Python学习笔记1-数据类型

    数据类型: float — 浮点数可以精确到小数点后面15位 int — 整型可以无限大 bool — 非零为true,零为false list — 列表 Float/Int: 运算符: / — 浮点 ...

  9. linux top 源码分析

    /* * Copyright (c) 2008, The Android Open Source Project * All rights reserved. * * Redistribution a ...

  10. HTML基本组成结构与标签的认识

    HTML基本组成结构与标签 其实组成结构用一张图来简单了解下如下 目前一般网站的结构是会如此不是很清晰简单 先来说说header头部 这样是不是更加清楚了 导航栏是引导用户查看网站内容的快捷入口,打个 ...