求助,ValueError: View function did not return a response

2017-03-01 23:56:00 +08:00
 BMYY

新手在学 flask 开发 T.T ,看到《 python web 开发》第十章的用户资料, 在搞 管理员级别的资料编辑器 那里出问题了, 视图函数抛出异常,不能返回响应 。百度,谷歌, sof 无果。找好久了没解决,心塞 所以来寻求帮助,谢谢能提供帮助的人。 这是整个文件夹 http://pan.baidu.com/s/1kV4sXcr

ValueError
ValueError: View function did not return a response

Traceback (most recent call last)
File "D:\web_develop\venv\lib\site-packages\flask-0.12-py2.7.egg\flask\app.py", line 1994, in __call__
return self.wsgi_app(environ, start_response)
File "D:\web_develop\venv\lib\site-packages\flask-0.12-py2.7.egg\flask\app.py", line 1985, in wsgi_app
response = self.handle_exception(e)
File "D:\web_develop\venv\lib\site-packages\flask-0.12-py2.7.egg\flask\app.py", line 1540, in handle_exception
reraise(exc_type, exc_value, tb)
File "D:\web_develop\venv\lib\site-packages\flask-0.12-py2.7.egg\flask\app.py", line 1982, in wsgi_app
response = self.full_dispatch_request()
File "D:\web_develop\venv\lib\site-packages\flask-0.12-py2.7.egg\flask\app.py", line 1615, in full_dispatch_request
return self.finalize_request(rv)
File "D:\web_develop\venv\lib\site-packages\flask-0.12-py2.7.egg\flask\app.py", line 1630, in finalize_request
response = self.make_response(rv)
File "D:\web_develop\venv\lib\site-packages\flask-0.12-py2.7.egg\flask\app.py", line 1725, in make_response
raise ValueError('View function did not return a response')
ValueError: View function did not return a response

##路由视图 views.py

@main.route('/edit-profile/<int:id>', methods=['GET', 'POST'])
@login_required
@admin_required
def edit_profile_admin(id):
	user = User.query.get_or_404(id)
	form = EditProfileAdminForm(user=user)
	if form.validate_on_submit():
		user.email = form.email.data
		user.username = form.username.data
		user.confirmed = form.confirmed.data
		user.role = Role.query.get(form.role.data)
		user.name = form.name.data
		user.location = form.location.data
		user.about_me = form.about_me.data
		db.session.add(user)
		flash('The profile has been updated.')
		return redirect(url_for('.user', username=user.username))
	form.email.data = user.email
	form.username.data = user.username
	form.confirmed.data = user.confirmed
	form.role.data = user.role_id
	form.name.data = user.name
	form.location.data = user.location
	form.about_me.data = user.about_me
	return render_template('edit_profile.html', form=form, user=user)

##表单 forms.py:

class EditProfileAdminForm(FlaskForm):
	email = StringField('Email', validators=[Required(), Length(1, 64),
		Email()])
	username = StringField('Username', validators=[
		Required(), Length(1, 64), Regexp('^[A-Za-z][A-Za-z0-9_.]*$', 0,
										  'Username must have only letters,'
										   'number, dots or underscores')])
	confirmed = BooleanField('Confirmed')
	role = SelectField('Role', coerce=int)
	name = StringField('Real name', validators=[Length(0, 64)])
	location = StringField('Location', validators=[Length(0, 64)])
	about_me = TextAreaField('About me')
	submit = SubmitField('Submit')
	
	def __init__(self, user, *args, **kwargs):
		super(EditProfileAdminForm, self).__init__(*args, **kwargs)
		self.role.choices = [(role.id, role.name)
							 for role in Role.query.order_by(Role.name).all()]
		self.user = user
		
	def validate_email(self, field):
		if field.data != self.user.email and \
				User.query.filter_by(email=field.date).first():
			raise ValidationError('Emai already registered.')
	
	def validate_username(self, field):
		if field.data !=self.user.username and \
				User.query.filter_by(username=field.data).first():
			raise ValidationError('Username already in use.')

##模板 edit_profile.html :

{% extends "base.html" %}
{% import "bootstrap/wtf.html" as wtf %}

{% block title %}Flasky - Edit Profile{% endblock %}

{% block page_content %}
<div class="page-header">
	<h1>Edit Your Profile</h1>
</div>
<div class="col-md-4">
	{{ wtf.quick_form(form) }}
</div>
{% endblock %}
4663 次点击
所在节点    Python
5 条回复
fy
2017-03-02 00:13:15 +08:00
没渲染到模板?楼主试试 ret = render_template('edit_profile.html', form=form, user=user)

然后 print 看看 ret 里头是啥再 return

可能是路径配错了。

另外貌似楼主所有的缩进都是 TAB ,而不是空格。
flaneurse
2017-03-02 02:47:34 +08:00
为啥不去作者 github 问
BMYY
2017-03-02 10:09:17 +08:00
@fy 谢谢你的回答
试了你的方法抛出错误一样,没 print 出啥东西。

其他网友找到问题了,检查用户权限的自定义装饰器出问题了 decorators.py

```
def permission_required(permission):
def decorator(f):
@wraps(f)
def decorated_function(*args, **kwargs):
if not current_user.can(permission):
abort(403)
return f(*args, **kwargs) # 此行缩进错误,应在 if 语句外
return decorated_function
return decorator
```

修改过来就可以了。

T.T ,真是自己粗心大意。
python 代码缩进问题,我特意去查了,网友都建议缩进用 4 个空格,用 1 个 tab 代替风险太大,因为各家的编辑器对 tab 键定义存在差异,很容易出问题。虽然说我这问题不是出在 tab 上,之后我还是会注意这个问题的。感谢。

不过我还有一问题,为什么错误页面抛出的是 视图函数 View function 的问题,
而不会抛出装饰器那边的错误,例如( IndentationError ):
@admin_required → def permission_required(permission)
谢谢
BMYY
2017-03-02 10:13:35 +08:00
@flaneurse
谢谢回答,因为我想如果能在这解决就快一点
如果不能解决再去 github 问的
用了你的建议上午确实也去作者 github 提交了提问 T.T
看来我要自答告诉作者解决了~
fy
2017-03-02 13:32:04 +08:00
@BMYY = = 没 print 出东西本身就已经说明了很多问题,要么是输出是 None 、'' 之类,要么是代码没执行。

调试的时候顺藤摸瓜向下找就是了。

关于你抛出异常的函数的疑问,原因是你的 View function 就是带上装饰器之后的函数。

而不是你的初始函数了。装饰器实际上产生了一个新的函数传入进去。

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/344205

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX