Eri*_*aal 23 python sqlalchemy flask flask-sqlalchemy
我有一个正在运行的Flask应用程序,它是根据我们在网上和Miguel Grinberg的"Flask Web Development"一书中找到的最佳实践组合而建立的.
我们现在需要第二个Python应用程序,它不是Web应用程序,需要访问与Flask应用程序相同的模型.我们想重复使用相同的模型,因此两个应用程序都可以从共享代码中受益.
我们已经删除了对flask-sqlalchemy扩展(我们之前使用过Flask应用程序时)的依赖关系.并将其替换为此处描述的SQLalchemy Declarative扩展,这有点简单(Flask-SQLalchemy为标准SQLAlchemy添加了一些特定的东西)
根据示例,我们在根目录中创建了一个database.py文件.在我们的例子中,有两个与Declarative扩展示例不同的东西:我将引擎和会话放在一个类中,因为我们所有的模型都使用db.session而不是db_session,并且我将带有配置值的字典传递给init( ),以便我可以使用不同的配置从Flask和其他应用程序重新使用此database.py.它看起来像这样:
from sqlalchemy import create_engine
from sqlalchemy.orm import scoped_session, sessionmaker
from sqlalchemy.ext.declarative import declarative_base
class Database(object):
def __init__(self, cfg):
self.engine = create_engine(cfg['SQLALCHEMY_DATABASE_URI'], convert_unicode=True)
self.session = scoped_session(sessionmaker(autocommit=False, autoflush=False, bind=self.engine))
class Model(object):
pass
Base = declarative_base()
Run Code Online (Sandbox Code Playgroud)
所以现在我们来解决实际问题.Flask创建一个包含配置选项的类字典对象,并将它们作为属性添加到app-instance.它从实例文件夹,站点根目录中的config.py和环境变量加载它们.我需要从Flask传递配置字典,所以我需要Flask来FIRST加载和组装配置,然后初始化数据库,并在app文件的根目录中有一个(配置的)db对象.但是,我们遵循应用工厂模式,因此我们可以针对不同情况(测试,生产,开发)使用不同的配置.
这意味着我们app/__init__.py看起来像这样(简化):
from flask import Flask
from database import Database
from flask.ext.mail import Mail
from flask_bcrypt import Bcrypt
from config import config
mail = Mail()
bcrypt = Bcrypt()
def create_app(config_name):
app = Flask(__name__, instance_relative_config=True)
if not config_name:
config_name = 'default'
app.config.from_object(config[config_name])
app.config.from_pyfile('config.py')
config[config_name].init_app(app)
db = Database(app.config)
mail.init_app(app)
bcrypt.init_app(app)
@app.teardown_appcontext
def shutdown_session(exception=None):
db.session.remove()
from main import main as main_blueprint
app.register_blueprint(main_blueprint)
return app
Run Code Online (Sandbox Code Playgroud)
但是db(模型从..导入)现在需要在create_app()函数内部,因为这是Flask加载配置的地方.如果我要在create_app()函数之外实例化db对象,它可以从模型导入,但是没有配置!
示例模型看起来像这样,正如您所看到的,它期望应用程序根目录中的"db":
from . base_models import areas
from sqlalchemy.orm import relationship, backref
from ..utils.helper_functions import newid
from .. import db
class Areas(db.Model, areas):
"""Area model class.
"""
country = relationship("Countries", backref=backref('areas'))
def __init__(self, *args, **kwargs):
self.area_id = newid()
super(Areas, self).__init__(*args, **kwargs)
def __str__(self):
return u"{}".format(self.area_name).encode('utf8')
def __repr__(self):
return u"<Area: '{}'>".format(self.area_name).encode('utf8')
Run Code Online (Sandbox Code Playgroud)
所以我的问题是,我怎样才能有一个可以在外部配置的数据库实例(通过Flask或其他应用程序),并仍然使用应用程序工厂模式?
编辑:代码示例不正确,它有一个导入Flask-SQLalchemy被替换为from database import Database.对不起任何困惑.
dav*_*ism 20
与大多数Flask扩展一样,Flask-SQLAlchemy扩展应该在工厂外创建,然后在工厂中使用初始化init_app.这样您就可以db在创建应用程序之前使用该对象.
db = SQLAlchemy()
def create_app():
app = Flask(__name__)
db.init_app(app)
return app
Run Code Online (Sandbox Code Playgroud)
与任何设计合理的Python项目一样,您的Flask应用程序应该是可安装的程序包.这很简单:确保项目布局有意义,然后添加基本setup.py文件.
project/
my_flask_package/
__init__.py # at the most basic, this contains create_app and db
setup.py
Run Code Online (Sandbox Code Playgroud)
from setuptools import setup, find_packages
setup(
name='my_flask_package',
version='1.0',
packages=find_packages(),
install_requires=['flask', 'flask-sqlalchemy'],
)
Run Code Online (Sandbox Code Playgroud)
$ python setup.py sdist
Run Code Online (Sandbox Code Playgroud)
现在,您可以安装Flask应用程序及其数据库,以便在其他项目中使用.在第二个项目的virtualenv中安装并导入它,然后创建并推送一个应用程序来初始化它.
$ pip install my_flask_package-1.0.tar.gz
Run Code Online (Sandbox Code Playgroud)
from my_flask_package import db, create_app
create_app().app_context().push()
db.session.query(...)
Run Code Online (Sandbox Code Playgroud)
如果您担心创建应用程序所涉及的开销,可以向create_app函数添加参数以控制初始化的内容.对于大多数情况,这不应该是一个问题.
| 归档时间: |
|
| 查看次数: |
7097 次 |
| 最近记录: |