AttributeError:'NoneType'对象没有属性'app'

时间:2013-06-20 06:31:44

标签: python flask xhtml2pdf

以下代码给出错误:

Traceback (most recent call last):
  File "pdf.py", line 14, in <module>
    create_pdf(render_template('templates.htm'))
  File "/usr/local/lib/python2.7/dist-packages/flask/templating.py", line 123, in render_template
    ctx.app.update_template_context(context)
AttributeError: 'NoneType' object has no attribute 'app'

代码:

from xhtml2pdf import pisa
from StringIO import StringIO
from flask import render_template,Flask

app=Flask(__name__)
app.debug=True

@app.route("/")
def create_pdf(pdf_data):
        filename= "file.pdf"
        pdf=pisa.CreatePDF( StringIO(pdf_data),file(filename, "wb"))

if __name__ == "__main__":
        create_pdf(render_template('templates.htm'))

5 个答案:

答案 0 :(得分:15)

Martin的回答很好地解释了为什么发生此错误。

接受的答案解决了所提出的问题,但肯定不是唯一的方法。就我而言,我有更喜欢的东西:

import threading

from flask import Flask, render_template

app = Flask("myapp")

app.route('/')
def get_thing(thing_id):
    thing = cache.get(thing_id)
    if thing is None:
        # Handle cache miss...
    elif is_old(thing):
        # We'll serve the stale content but let's
        # update the cache in a background thread
        t = threading.Thread(
            target=get_thing_from_datastore_render_and_cache_it,
            args=(thing_id,)
        )
        t.start()
    return thing

def get_thing_from_datastore_render_and_cache_it(thing_id):
    thing = datastore.get(thing_id)
    cache.set(render_template(thing))

但是当在Flask请求周期之外的后台线程中运行get_thing_from_datastore_render_and_cache_it时,我收到上面显示的错误,因为该线程无法访问请求上下文。

发生错误是因为Flask提供了一个允许自动访问模板中的请求变量的开发人员快捷方式 - 换句话说,这是由Flask关于如何包装Jinja2的功能而不是Jinja2本身的决定引起的。我解决这个问题的方法就是直接使用Jinja2的渲染:

import jinja2

def render_without_request(template_name, **template_vars):
    """
    Usage is the same as flask.render_template:

    render_without_request('my_template.html', var1='foo', var2='bar')
    """
    env = jinja2.Environment(
        loader=jinja2.PackageLoader('name.ofmy.package','templates')
    )
    template = env.get_template(template_name)
    return template.render(**template_vars)

该函数假定您的Flask应用程序具有传统模板子文件夹。具体来说,这里的项目结构将是

.
└── name/
    ├── ofmy/
    |   ├── package/
    |   |   ├── __init__.py <--- Where your Flask application object is defined
    |   |   └── templates/
    |   |       └── my_template.html
    |   └── __init__.py
    └── __init__.py

如果templates/下有子目录结构,则只需传递模板文件夹根目录的相对路径,就像使用Flask的render_template时一样。

答案 1 :(得分:12)

Flask做了很多'魔术',所以你不必担心路由或解析请求。当Flask应用程序收到请求时,它会在将逻辑委托给您的视图函数之前创建一个“上下文”对象。

在您的代码中,您直接调用render_template而不通过Flask,因此不会创建上下文。 render_template尝试通过此上下文(app)访问您的应用程序(ctx),即None,因此错误:

AttributeError: 'NoneType' object has no attribute 'app'

现在这不是你的代码唯一的错误。查看函数(使用装饰器@app.route(...)注册)不应直接调用。 @ rajpy的回答为您提供了如何使用它们的一个很好的例子。

答案 2 :(得分:4)

从代码中,我可以看到您希望允许用户下载pdf。

from xhtml2pdf import pisa
from StringIO import StringIO
from flask import render_template,Flask, Response

app=Flask(__name__)
app.debug=True

@app.route("/")
def create_pdf(pdf_data):
        filename= "file.pdf"
        pdf=pisa.CreatePDF( StringIO(pdf_data),file(filename, "wb"))
        return Response(pdf, mimetype='application/octet-stream',
                        headers={"Content-Disposition": "attachment;filename=%s" % filename})

if __name__ == "__main__":
        app.run()

现在,运行python aboveprogram.py

转到http://localhost:5000

浏览器提示下载PDF。 希望它有所帮助..

答案 3 :(得分:4)

当尝试从Celery任务渲染模板时,我遇到了同样的问题。

最简单的解决方案是manually push所需的上下文:

with app.app_context():
    # Code calling render_template goes here

答案 4 :(得分:0)

我的问题(在芹菜任务中)通过在应用上下文中呈现模板来解决。

with app.app_context(), app.test_request_context():
    template = render_template('home')

有关说明,请转到HERE