首页
学习
活动
专区
工具
TVP
发布
社区首页 >问答首页 >Flask -在应用程序上下文之外工作

Flask -在应用程序上下文之外工作
EN

Stack Overflow用户
提问于 2016-03-25 10:59:54
回答 2查看 10.1K关注 0票数 2

我正在测试一个Flask应用程序,并收到一个“在应用程序上下文之外工作”的错误。我的文件目录如下:

代码语言:javascript
复制
api
    app.py
    __init__.py
    models
        __init__.py
        user.py
    resources
        __init__.py
        deals.py
        stores.py
    common
        __init__.py
        calculations.py
        decorators.py

我的app.py文件如下所示:

代码语言:javascript
复制
import os
from flask import Flask, jsonify, url_for, redirect, request, g, current_app
from flask_pymongo import PyMongo
from flask_restful import Api, Resource
from flask_httpauth import HTTPTokenAuth
from flask.ext.sqlalchemy import SQLAlchemy
from flask.ext.httpauth import HTTPBasicAuth

from resources.deals import Deals
from resources.stores import Stores

from models.user import User

USERDBFILE=os.path.join(os.path.join(os.path.dirname(os.path.realpath(__file__)),'database'),'db.sqlite')

#Deals database
app = Flask(__name__)
app.config["MONGO_DBNAME"] = "database"
mongo = PyMongo(app,config_prefix='MONGO')
app.db = mongo

#User database
app.config['SECRET_KEY'] = 'SECRET KEY'
app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///db.sqlite'
app.config['SQLALCHEMY_COMMIT_ON_TEARDOWN'] = True
app.dbuser = SQLAlchemy(app)

#App url
app.APP_URL = "http://127.0.0.1:5000"

#Setup authorization
auth = HTTPTokenAuth(scheme='Token')

#Setup the app
api = Api(app)
api.add_resource(Deals, '/deals', '/Deals/<string:type>/<string:id>',endpoint="dealType")
api.add_resource(Stores, '/stores', '/Stores/<string:type>/<string:id>',endpoint="type")

if __name__ == "__main__":
    if not os.path.exists(USERDBFILE):
        app.dbuser.create_all()
    app.run(debug=True)

我的users.py文件如下:

代码语言:javascript
复制
from flask import current_app
import os
from flask import Flask, abort, request, jsonify, g, url_for
from flask.ext.sqlalchemy import SQLAlchemy
from flask.ext.httpauth import HTTPBasicAuth
from passlib.apps import custom_app_context as pwd_context
from itsdangerous import (TimedJSONWebSignatureSerializer
                          as Serializer, BadSignature, SignatureExpired)

class User(current_app.dbuser.Model):
    __tablename__ = 'user_api'
    id = current_app.dbuser.Column(current_app.dbuser.Integer,primary_key=True)
    date_created = current_app.dbuser.Column(current_app.dbuser.DateTime,default=current_app.dbuser.func.current_timestamp())
    date_modified = current_app.dbuser.Column(current_app.dbuser.DateTime,default=current_app.dbuser.func.current_timestamp(),
                                      onupdate=current_app.dbuser.func.current_timestamp())
    # User Name
    name = current_app.dbuser.Column(current_app.dbuser.String(128),nullable=False)
    # Identification Data: email & password
    email = current_app.dbuser.Column(current_app.dbuser.String(128),nullable=False,unique=True)
    password = current_app.dbuser.Column(current_app.dbuser.String(192),nullable=False)
    company = current_app.dbuser.Column(current_app.dbuser.String(128),nullable=False,unique=True)
    # Authorization Data: role & status
    role = current_app.dbuser.Column(current_app.dbuser.String(32),nullable=False,default='user')
    status = current_app.dbuser.Column(current_app.dbuser.Boolean,nullable=False,default=True)
    hourly_limit = current_app.dbuser.Column(current_app.dbuser.Integer,nullable=False,default=100)
    daily_limit = current_app.dbuser.Column(current_app.dbuser.Integer,nullable=False,default=2400)
    monthly_limit = current_app.dbuser.Column(current_app.dbuser.Integer,nullable=False,default=2400)
    admin = current_app.dbuser.Column(current_app.dbuser.Boolean,nullable=False,default=True)

    def hash_password(self, password):
        self.password_hash = pwd_context.encrypt(password)

    def verify_password(self, password):
        return pwd_context.verify(password, self.password_hash)

    def generate_auth_token(self, expiration=600):
        s = Serializer(current_app.config['SECRET_KEY'], expires_in=expiration)
        return s.dumps({'id': self.id})

    @staticmethod
    def verify_auth_token(token):
        s = Serializer(current_app.config['SECRET_KEY'])
        try:
            data = s.loads(token)
        except SignatureExpired:
            return None    # valid token, but expired
        except BadSignature:
            return None    # invalid token
        user = User.query.get(data['id'])
        return user

我在与app.py相同的目录中运行该文件,使用

代码语言:javascript
复制
python app.py

但它返回以下错误:

代码语言:javascript
复制
File "app.py", line 13, in <module>
    from models.user import User
  File "/Users/toby/api/api/models/user.py", line 10, in <module>
    class User(current_app.dbuser.Model):
  File "/Users/toby/api/venv/lib/python3.4/site-packages/werkzeug/local.py", line 343, in __getattr__
    return getattr(self._get_current_object(), name)
  File "/Users/toby/api/venv/lib/python3.4/site-packages/werkzeug/local.py", line 302, in _get_current_object
    return self.__local()
  File "/Users/toby/api/venv/lib/python3.4/site-packages/flask/globals.py", line 34, in _find_app
    raise RuntimeError('working outside of application context')
RuntimeError: working outside of application context

如果我将user.py文件的内容移到app.py文件中,并将继承从current_app.dbuser.Model更改为app.dbuser.Model,似乎可以很好地工作。有人知道我做错了什么吗?

EN

回答 2

Stack Overflow用户

发布于 2016-03-28 21:43:20

Flask-Sqlalchemy将会话、引擎和解密基础等sqlalchemy概念绑定到flask应用程序。这很方便,因为您只需要在uwsgi入口点实例化一件事( app对象),但在测试时却很痛苦-因为您必须实例化app对象。

编辑-我离开了下面关于测试的部分,但我重新阅读了你的问题,并意识到你实际上并不是在尝试测试任何东西。

在导入时(当您尝试初始化sqlalchemy模型时),您没有访问“current_app”对象的权限。相反,您必须实际从应用程序文件中导入应用程序对象。这当然意味着你必须担心循环依赖……

我有一个名为'register_routes‘的方法,它在我初始化app对象后被调用,该对象导入需要在导入时访问app对象的模型和视图文件。

代码语言:javascript
复制
#at the bottom of app.py
def register_models(app):
    from models import User

register_models(app)
代码语言:javascript
复制
# in models.py
from app import app

class User(app.dbuser.Model):
   ...

编辑-下面讨论关于单元测试的这个问题

Flask-Testing是一个试图解决这些问题的项目,几乎可以肯定的是,它适合这个领域的初学者-它提供了一个可以继承的测试类,它将在测试用例之前设置你的flask应用程序,并在测试用例之后拆除它。(当你开始了解各种全局变量以及它们的作用时,你可能想要离开这个……但它对入门非常有帮助!)

如果你不想这样做,你需要在使用flask-sqlalchemy模型之前创建一个应用程序并初始化一个应用程序上下文。这可能只是

代码语言:javascript
复制
app = myapp.create()
with app.test_request_context():
    # do some testing...

您可能希望在方法之间刷新此状态,否则全局状态将在测试用例之间泄漏。

票数 0
EN

Stack Overflow用户

发布于 2016-03-28 21:28:51

基本上,flask使用了相当多的全局变量,如current_apprequest等,这些变量只有在flask应用程序实例化并运行并处于各种状态时才存在。

您已经在用户对象的定义中使用了current_app,只要Python导入文件,就会计算该对象的值。您需要确保仅在应用程序已经运行时才使用这样的值。

您可以将User类的实例化移动到应用程序存在之后,但我认为根本问题是为什么要使用current_app.dbuser.Boolean而不是sqlalchemy.types.Boolean

我不是一个很好的flask.ext.sqlalchemy专家,但我猜你不需要从你拥有的应用程序的特定实例中加载ColumnBoolean之类的东西的定义。使用来自sqlalchemy的静态定义将阻止您拥有从User类到应用程序的依赖。

票数 -1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/36213353

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档