使用create_app()
功能时,我无法启动应用程序。我刚开始以这种方式构建应用程序,而且从我的所有研究中看来,我的方法似乎不同,因为我使用自己的数据库包装而不是SQLAlchemy - 这使得它很容易,因为可以使用db.init_app(app)
。
我的问题是:我似乎无法在/models/user.py中访问我的数据库连接...如何修复此问题以便我可以在该文件中使用数据库连接?
这是我的应用程序的文件夹结构,后面是列出的文件:
/api
/common
database.py
/models
user.py
/resources
user.py
app.py
run.py
这是我的文件
#
# File: run.py
#
from api.app import create_app
app = create_app(debug=True)
app.run(
host=app.config['APP_HOST'],
port=app.config['APP_PORT'],
debug=app.config['APP_DEBUG_FLASK'],
ssl_context=app.config['APP_SSL_CONTEXT']
)
#
# File: app.py
#
from logging.config import dictConfig
from flask import Flask
from flask_restful import Api
from api.config import LocalConfig, LiveConfig
from api.extensions import bcrypt, cors, jwt
from api.resources.user import *
from api.common.database import Database
def create_app(debug=True):
config = LocalConfig if debug else LiveConfig
# Create app
app = Flask(__name__)
# Set configuration variables
app.config.from_object(config)
app.secret_key = app.config['APP_SECRET_KEY']
app.url_map.strict_slashes = False
# Create api
api = Api(app, prefix='/api/v2')
# Initializing the logger
dictConfig(app.config['LOGGING'])
# Connect to mysql
db = Database(
host=app.config['MYSQL_HOST'],
db=app.config['MYSQL_DB'],
user=app.config['MYSQL_USER'],
passwd=app.config['MYSQL_PASS'],
)
register_decorators(app)
register_extensions(app)
register_endpoints(api)
return app
def register_extensions(app):
bcrypt.init_app(app)
jwt.init_app(app)
def register_endpoints(api):
api.add_resource(UserLogin, '/login')
#
# File: /resources/user.py
#
from flask_restful import Resource, reqparse
from api.models.user import *
class UserLogin(Resource):
def __init__(self):
self.reqparse = reqparse.RequestParser()
self.reqparse.add_argument('username', type=str, required=True,
help='Username is required.',
location='json')
self.reqparse.add_argument('password', type=str, default='', location='json')
def post(self):
args = self.reqparse.parse_args()
print(args['username'])
user = UserModel.get_by_username(args['username'])
return {'message': 'Wrong credentials'}
#
# File: /models/user.py
#
import datetime
import json
import logging
from api.common.database import Database
class UserModel:
@classmethod
def get_by_username(cls, username=None):
user = cls.db.getOne(
table='users',
fields=['user_id','data'],
where=('username = %s', [username])
)
if user:
user['data'] = json.loads(user['data'])
return user
#
# File: /common/database.py
#
import MySQLdb
class Database:
conn = None
cur = None
conf = None
def __init__(self, **kwargs):
self.conf = kwargs
self.conf['keep_alive'] = kwargs.get('keep_alive', False)
self.conf['charset'] = kwargs.get('charset', 'utf8')
self.conf['host'] = kwargs.get('host', 'localhost')
self.conf['port'] = kwargs.get('port', 3306)
self.conf['autocommit'] = kwargs.get('autocommit', False)
self.conf['ssl'] = kwargs.get('ssl', False)
self.connect()
def connect(self):
try:
if not self.conf['ssl']:
self.conn = MySQLdb.connect(db=self.conf['db'],
host=self.conf['host'],
port=self.conf['port'],
user=self.conf['user'],
passwd=self.conf['passwd'],
charset=self.conf['charset'])
else:
self.conn = MySQLdb.connect(db=self.conf['db'],
host=self.conf['host'],
port=self.conf['port'],
user=self.conf['user'],
passwd=self.conf['passwd'],
ssl=self.conf['ssl'],
charset=self.conf['charset'])
self.cur = self.conn.cursor(MySQLdb.cursors.DictCursor)
self.conn.autocommit(self.conf['autocommit'])
except:
print ('MySQL connection failed')
raise
def getOne(self, table=None, fields='', where=None, order=None, limit=(1,)):
### code that handles querying database directly ###
答案 0 :(得分:2)
我开始转向使用Miguel Grinberg在Flask Mega-Tutorial第一部分中插图的create_app
模式的形式。
在您的情况下,db
对象的引用锁定在create_app
内的局部变量中。诀窍是让它可见。考虑一下我在SQLAlchemy中使用的这个方案,你可以适应它使用你的包装器:
main.py
config.py
tests.py
app/
__init__.py
首先,默认配置看起来像这样(为了简洁而修剪)
# config.py
...
class Config:
TESTING = False
SQLALCHEMY_DATABASE_URI = ...
...
配置将在出厂时用作默认值。
# app/__init__.py
...
db = SQLAlchemy() # done here so that db is importable
migrate = Migrate()
def create_app(config_class=Config):
app = Flask(__name__)
app.config.from_object(config_class)
db.init_app(app)
migrate.init_app(app, db)
... register blueprints, configure logging etc.
return app
请注意from app import db
有效。
# main.py
from app import create_app
app = create_app()
然后FLASK_APP=main.py venv/bin/flask run
。
为了进行测试,Config的子类使用内存数据库(并进行其他调整以避免命中外部服务)。
# tests.py
...
class TestConfig(Config):
TESTING = True
SQLALCHEMY_DATABASE_URI = 'sqlite://'
class ExampleTests(unittest.TestCase):
def setUp(self):
self.app = create_app(TestConfig)
# See Grinberg's tutorial for the other essential bits