《Flask Web开发》学习笔记之bug--(4)【AttributeError: type object 'datetime.datetime' has no attribute 'utnow'】

在学习《Flask Web开发》过程中,遇到了一个Python bug,具体表现为尝试调用datetime.datetime的'utnow'方法时,报错AttributeError,指出该类型对象没有'utnow'属性。本文详细记录了这个错误的原因分析和解决过程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

from datetime import datetime
from flask import Flask, render_template
from flask_script import Manager
from flask_bootstrap import Bootstrap
from flask_moment import Moment
from flask_wtf import Form
from wtforms import StringField, SubmitField
from wtforms.validators import Required

class NameForm(Form):
	name = StringField('What is your name?', validators=[Required])
	submit = SubmitField('Submit')
		

app = Flask(__name__)

app.config['SECRET_KEY'] = 'hard to guess string'
manager = Manager(app)
bootstrap = Bootstrap(app)
moment = Moment(app)

@app.errorhandler(404)
def page_not_found(e):
	return render_template('404.html'), 404

@app.errorhandler(500)
def internal_server_error(e):
	return render_template('500.html'), 500

@app.route('/')
def index():
	return render_template('index.html', current_time=datetime.utnow())

@app.route('/user/<name>')
def user(name):
	return render_template('user.html', name=name)


if __name__ == '__main__':
	manager.run()


$ python3 hello.py runserver --host 0.0.0.0

报错:

* Running on http://0.0.0.0:5000/ (Press CTRL+C to quit)
[2017-07-13 14:05:19,132] ERROR in app: Exception on / [GET]
Traceback (most recent call last):
  File "/home/henry/.local/lib/python3.5/site-packages/flask/app.py", line 1982, in wsgi_app
    response = self.full_dispatch_request()
  File "/home/henry/.local/lib/python3.5/site-packages/flask/app.py", line 1614, in full_dispatch_request
    rv = self.handle_user_exception(e)
  File "/home/henry/.local/lib/python3.5/site-packages/flask/app.py", line 1517, in handle_user_exception
    reraise(exc_type, exc_value, tb)
  File "/home/henry/.local/lib/python3.5/site-packages/flask/_compat.py", line 33, in reraise
    raise value
  File "/home/henry/.local/lib/python3.5/site-packages/flask/app.py", line 1612, in full_dispatch_request
    rv = self.dispatch_request()
  File "/home/henry/.local/lib/python3.5/site-packages/flask/app.py", line 1598, in dispatch_request
    return self.view_functions[rule.endpoint](**req.view_args)
  File "hello.py", line 32, in index
    return render_template('index.html', current_time=datetime.utnow())
AttributeError: type object 'datetime.datetime' has no attribute 'utnow'
127.0.0.1 - - [13/Jul/2017 14:05:19] "GET / HTTP/1.1" 500 -
Error on request:
Traceback (most recent call last):
  File "/home/henry/.local/lib/python3.5/site-packages/werkzeug/serving.py", line 209, in run_wsgi
    execute(self.server.app)
  File "/home/henry/.local/lib/python3.5/site-packages/werkzeug/serving.py", line 197, in execute
    application_iter = app(environ, start_response)
  File "/home/henry/.local/lib/python3.5/site-packages/flask/app.py", line 1997, in __call__
    return self.wsgi_app(environ, start_response)
  File "/home/henry/.local/lib/python3.5/site-packages/flask/app.py", line 1985, in wsgi_app
    response = self.handle_exception(e)
  File "/home/henry/.local/lib/python3.5/site-packages/flask/app.py", line 1547, in handle_exception
    return self.finalize_request(handler(e), from_error_handler=True)
  File "/home/henry/dev/Flask_Web_Python_Henry/flask_henry/hello.py", line 28, in internal_server_error
    return render_template('500.html'), 500
  File "/home/henry/.local/lib/python3.5/site-packages/flask/templating.py", line 133, in render_template
    return _render(ctx.app.jinja_env.get_or_select_template(template_name_or_list),
  File "/home/henry/.local/lib/python3.5/site-packages/jinja2/environment.py", line 869, in get_or_select_template
    return self.get_template(template_name_or_list, parent, globals)
  File "/home/henry/.local/lib/python3.5/site-packages/jinja2/environment.py", line 830, in get_template
    return self._load_template(name, self.make_globals(globals))
  File "/home/henry/.local/lib/python3.5/site-packages/jinja2/environment.py", line 804, in _load_template
    template = self.loader.load(self, name, globals)
  File "/home/henry/.local/lib/python3.5/site-packages/jinja2/loaders.py", line 125, in load
    code = environment.compile(source, name, filename)
  File "/home/henry/.local/lib/python3.5/site-packages/jinja2/environment.py", line 591, in compile
    self.handle_exception(exc_info, source_hint=source_hint)
  File "/home/henry/.local/lib/python3.5/site-packages/jinja2/environment.py", line 780, in handle_exception
    reraise(exc_type, exc_value, tb)
  File "/home/henry/.local/lib/python3.5/site-packages/jinja2/_compat.py", line 37, in reraise
    raise value.with_traceback(tb)
  File "/home/henry/dev/Flask_Web_Python_Henry/flask_henry/templates/500.html", line 9, in template
    {% endlock %}
  File "/home/henry/.local/lib/python3.5/site-packages/jinja2/environment.py", line 497, in _parse
    return Parser(self, source, name, encode_filename(filename)).parse()
  File "/home/henry/.local/lib/python3.5/site-packages/jinja2/parser.py", line 896, in parse
    result = nodes.Template(self.subparse(), lineno=1)
  File "/home/henry/.local/lib/python3.5/site-packages/jinja2/parser.py", line 878, in subparse
    rv = self.parse_statement()
  File "/home/henry/.local/lib/python3.5/site-packages/jinja2/parser.py", line 130, in parse_statement
    return getattr(self, 'parse_' + self.stream.current.value)()
  File "/home/henry/.local/lib/python3.5/site-packages/jinja2/parser.py", line 268, in parse_block
    node.body = self.parse_statements(('name:endblock',), drop_needle=True)
  File "/home/henry/.local/lib/python3.5/site-packages/jinja2/parser.py", line 165, in parse_statements
    result = self.subparse(end_tokens)
  File "/home/henry/.local/lib/python3.5/site-packages/jinja2/parser.py", line 878, in subparse
    rv = self.parse_statement()
  File "/home/henry/.local/lib/python3.5/site-packages/jinja2/parser.py", line 144, in parse_statement
    self.fail_unknown_tag(token.value, token.lineno)
  File "/home/henry/.local/lib/python3.5/site-packages/jinja2/parser.py", line 97, in fail_unknown_tag
    return self._fail_ut_eof(name, self._end_token_stack, lineno)
  File "/home/henry/.local/lib/python3.5/site-packages/jinja2/parser.py", line 90, in _fail_ut_eof
    self.fail(' '.join(message), lineno)
  File "/home/henry/.local/lib/python3.5/site-packages/jinja2/parser.py", line 59, in fail
    raise exc(msg, lineno, self.name, self.filename)
jinja2.exceptions.TemplateSyntaxError: Encountered unknown tag 'endlock'. Jinja was looking for the following tags: 'endblock'. The innermost block that needs to be closed is 'block'.

注意到:


File "hello.py", line 32, in index
    return render_template('index.html', current_time=datetime.utnow())
AttributeError: type object 'datetime.datetime' has no attribute 'utnow'


原来是【utnow()】少了一个字母【c】,加上去了之后就没有这个报错了。






<think>我们遇到了一个AttributeError,指出在flask_wtf.csrf模块中没有'exempt'属性。 在Flask-WTF中,通常我们使用`csrf.exempt`来装饰视图函数以禁用CSRF保护。但是,根据错误信息,似乎在新版本中这个属性可能被移动或删除了。 实际上,在Flask-WTF 0.14.2之后,`flask_wtf.csrf`模块不再直接提供`exempt`属性。相反,我们应该使用`csrf.CSRFProtect`的实例来调用`exempt`方法。 正确的做法是: 1. 首先,我们通常会在应用上初始化CSRFProtect,如: csrf = CSRFProtect(app) 或者 csrf = CSRFProtect() csrf.init_app(app) 2. 然后,我们可以使用这个实例的exempt方法来装饰视图函数: @csrf.exempt def my_view(): ... 但是,错误提示表明我们尝试使用`flask_wtf.csrf.exempt`,这可能是旧版本的用法,或者导入方式不正确。 因此,解决方法是: - 确保我们有一个`CSRFProtect`的实例(比如名为`csrf`),然后使用这个实例的`exempt`方法。 - 或者,如果我们使用的是蓝图,也可以使用蓝图的豁免方法(如果使用Flask-WTF的特定版本支持的话)。 如果我们在某个地方有类似这样的代码: from flask_wtf.csrf import exempt 这在新版本中已经不可用了,应该改为: from flask_wtf import CSRFProtect csrf = CSRFProtect() # 然后使用@csrf.exempt 或者,如果我们已经有一个名为`csrf`的实例,那么直接使用它。 所以,错误修正:将导入方式改为使用CSRFProtect实例的exempt方法。 示例代码: from flask import Flask from flask_wtf.csrf import CSRFProtect app = Flask(__name__) csrf = CSRFProtect(app) @app.route('/some-route', methods=['POST']) @csrf.exempt # 使用我们创建的csrf实例的exempt方法 def some_route(): return "CSRF exempted" 注意:在旧版本中(如0.14.2之前),我们可能可以这样导入: from flask_wtf.csrf import csrf_exempt as exempt 但在新版本中,这个方式已经改变。 因此,请检查你的Flask-WTF版本,并更新代码以使用新的方式。 如果问题仍然存在,请检查是否在代码中错误地使用了`flask_wtf.csrf.exempt`,而应该使用`csrf.exempt`(其中`csrf`是`CSRFProtect`的实例)。 另外,还有一种可能是,你正在使用的扩展或代码片段是针对旧版本Flask-WTF编写的。 总结:将代码中所有使用`@flask_wtf.csrf.exempt`的地方改为使用`@csrf.exempt`,并确保`csrf`是`CSRFProtect`的一个实例。 如果是在蓝图中豁免CSRF,同样使用`@csrf.exempt`装饰蓝图中的视图函数。 注意:在蓝图中,如果要在整个蓝图禁用CSRF,可以使用: blueprint = Blueprint(...) csrf.exempt(blueprint) 但请注意,这需要Flask-WTF版本>=0.14.2(具体版本要求可能需查看文档)。 因此,请根据实际情况调整。</think>### 问题分析 出现 `AttributeError: module 'flask_wtf.csrf' has no attribute 'exempt'` 错误的原因是 Flask-WTF 0.15 版本后移除了直接导入 `csrf.exempt` 的方式。新版本需通过 `CSRFProtect` 实例调用 `exempt` 方法。 ### 解决方案 #### 正确用法(推荐) ```python from flask import Flask from flask_wtf.csrf import CSRFProtect app = Flask(__name__) csrf = CSRFProtect(app) # 初始化 CSRFProtect 实例 # 使用实例的 exempt 方法装饰视图 @app.route("/no-csrf") @csrf.exempt # 正确用法 def no_csrf_view(): return "CSRF Exempted" ``` #### 兼容旧版本(0.14.x 及之前) ```python # 旧版本兼容写法(不推荐) from flask_wtf.csrf import csrf_exempt # 直接导入装饰器 @app.route("/old-route") @csrf_exempt def old_view(): return "Legacy Exempt" ``` #### 其他注意事项 1. **检查 Flask-WTF 版本**: ```bash pip show flask-wtf ``` 若版本 ≥ 0.15,必须使用 `CSRFProtect` 实例。 2. **蓝图豁免**: ```python blueprint = Blueprint("my_blueprint", __name__) csrf.exempt(blueprint) # 整个蓝图豁免 ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值