这是我用Python的轻量级框架 Flask 编写web程序时的一些经验,我将它记录在此,本篇是一系列文章中的第五篇。
下面是这个系列中已经发布的文章的目录:
在前面的章节,我们开始配置我们将要用到的Flask扩展。为了登录系统,我们将使用两个扩展,Flask-Login 和 Flask-OpenID. 配置如下所示 (fileapp\__init__.py):
import os from flaskext.login import LoginManager from flaskext.openid import OpenID from config import basedir lm = LoginManager() lm.setup_app(app) oid = OpenID(app, os.path.join(basedir, 'tmp'))Flask-OpenID 扩展为了可以存储临时文件,需要一个临时文件夹路径。为此,我们提供了它的位置。
重访我们的用户模型
Flask-Login扩展需要在我们的User类里实现一些方法。除了这些方法以外,类没有被要求实现其它方法。
下面是我们的User类 (fileapp/models.py):
class User(db.Model): id = db.Column(db.Integer, primary_key = True) nickname = db.Column(db.String(64), unique = True) email = db.Column(db.String(120), unique = True) role = db.Column(db.SmallInteger, default = ROLE_USER) posts = db.relationship('Post', backref = 'author', lazy = 'dynamic') def is_authenticated(self): return True def is_active(self): return True def is_anonymous(self): return False def get_id(self): return unicode(self.id) def __repr__(self): return '<User %r>' % (self.name)
is_authenticated方法是一个误导性的名字的方法,通常这个方法应该返回True,除非对象代表一个由于某种原因没有被认证的用户。
is_active方法应该为用户返回True除非用户不是激活的,例如,他们已经被禁了。
is_anonymous方法应该为那些不被获准登录的用户返回True。
最后,get_id方法为用户返回唯一的unicode标识符。我们用数据库层生成唯一的id。
接下来我们要更新登录视图函数(fileapp/views.py):
from flask import render_template, flash, redirect, session, url_for, request, g from flaskext.login import login_user, logout_user, current_user, login_required from app import app, db, lm, oid from forms import LoginForm from models import User, ROLE_USER, ROLE_ADMIN @app.route('/login', methods = ['GET', 'POST']) @oid.loginhandler def login(): if g.user is not None and g.user.is_authenticated(): return redirect(url_for('index')) form = LoginForm() if form.validate_on_submit(): session['remember_me'] = form.remember_me.data return oid.try_login(form.openid.data, ask_for = ['nickname', 'email']) return render_template('login.html', title = 'Sign In', form = form, providers = app.config['OPENID_PROVIDERS'])
注意到我们导入了一些新的模块,其中有些后面会用到。
跟上个版本的变化很小。我们给视图函数添加了一个新的装饰器:oid.loginhandler。它告诉Flask-OpenID这是我们的登录视图函数。
在方法体的开头,我们检测是是否用户是已经经过登录认证的,如果是就重定向到index页面。这儿的思路是如果一个用户已经登录了,那么我们不会让它做二次登录。
全局变量g是Flask设置的,在一个request生命周期中,用来存储和共享数据的变量。所以我猜你已经想到了,我们将把已经登录的用户放到g变量里。
我们在调用redirect()时使用的url_for()方法是Flask定义的从给定的view方法获取url。如果你想重定向到index页面,你h很可能使用redirect('/index'),但是我们有很好的理由让Flask为你构造url。
当我们从登录表单得到返回数据,接下来要运行的代码也是新写的。这儿我们做两件事。首先我们保存remember_me的布尔值到Flask的session中,别和Flask-SQLAlchemy的db.session混淆了。我们已经知道在一个request的生命周期中用Flask的g对象来保存和共享数据。沿着这条线路Flask的session提供了更多,更复杂的服务。一旦数据被保存到session中,它将在同一客户端发起的这次请求和这次以后的请求中永存而不会消亡。数据将保持在session中直到被明确的移除。为了做到这些,Flask为每个客户端建立各自的session。
这是我们实现的after_login方法(app/views.py)
@oid.after_login def after_login(resp): if resp.email is None or resp.email == "": flash('Invalid login. Please try again.') redirect(url_for('login')) user = User.query.filter_by(email = resp.email).first() if user is None: nickname = resp.nickname if nickname is None or nickname == "": nickname = resp.email.split('@')[0] user = User(nickname = nickname, email = resp.email, role = ROLE_USER) db.session.add(user) db.session.commit() remember_me = False if 'remember_me' in session: remember_me = session['remember_me'] session.pop('remember_me', None) login_user(user, remember = remember_me) return redirect(request.args.get('next') or url_for('index'))
传给after_login方法的resp参数包含了OpenID provider返回的一些信息。
第一个if声明仅仅是为了验证。我们要求一个有效的email,所以一个没有没提供的email我们是没法让他登录的。
评论删除后,数据将无法恢复
评论(15)
引用来自“yauchiang”的评论
File "E:\Python\microblog\app\views.py", line 38, in loginif g.user is not None and g.user.is_authenticated():
TypeError: 'bool' object is not callable
这是什么问题呢?我查了http://my.oschina.net/longfirst/blog/355621,g.user.is_authenticated()这个方法返回一个bool值啊?
引用来自“yauchiang”的评论
File "E:\Python\microblog\app\views.py", line 38, in loginif g.user is not None and g.user.is_authenticated():
TypeError: 'bool' object is not callable
这是什么问题呢?我查了http://my.oschina.net/longfirst/blog/355621,g.user.is_authenticated()这个方法返回一个bool值啊?
引用来自“楠哥哥大爱小Aom”的评论
同问引用来自“panweishadow”的评论
flask_login 0.3之后将authenticated从函数更改为属性,你看看你的login版本。参考:http://stackoverflow.com/questions/26313325/typeerror-bool-object-is-not-callable-in-flask引用来自“yauchiang”的评论
File "E:\Python\microblog\app\views.py", line 38, in loginif g.user is not None and g.user.is_authenticated():
TypeError: 'bool' object is not callable
这是什么问题呢?我查了http://my.oschina.net/longfirst/blog/355621,g.user.is_authenticated()这个方法返回一个bool值啊?
引用来自“楠哥哥大爱小Aom”的评论
同问引用来自“yauchiang”的评论
File "E:\Python\microblog\app\views.py", line 38, in loginif g.user is not None and g.user.is_authenticated():
TypeError: 'bool' object is not callable
这是什么问题呢?我查了http://my.oschina.net/longfirst/blog/355621,g.user.is_authenticated()这个方法返回一个bool值啊?
if g.user is not None and g.user.is_authenticated():
TypeError: 'bool' object is not callable
这是什么问题呢?我查了http://my.oschina.net/longfirst/blog/355621,g.user.is_authenticated()这个方法返回一个bool值啊?
def show_album(slug):
r= get_root_dirs_files()
try:
r = r[slug]
except:
return 'Album Not Existing'
return render_template("albums.html", dicts=r)
@app.route("/delete/<path:object_name>")
def delete(object_name):
obj = storage.get(object_name)
print obj
remove('./static/uploads/Pictures/%s' % (obj.name))
return redirect(url_for('show_album'???))
delete函数中,url_for show_album 函数时,里面的参数要怎么填?
from flask.ext.login import LoginManager
from flask.ext.openid import OpenID