Я действительно изо всех сил пытаюсь получить правильную настройку для Flask, SQLAlchemy и Celery. Я много искал и пробовал разные подходы, похоже, ничего не работает. Либо я пропустил контекст приложения, либо не могу запустить рабочих, либо есть другие проблемы. Структура очень общая, так что я могу создать приложение большего размера.
Я использую: Flask 0.10.1, SQLAlchemy 1.0, Celery 3.1.13, мои текущие настройки следующие:
Приложение /__ init__.py
#Empty
Приложение /config.py
import os
basedir = os.path.abspath(os.path.dirname(__file__))
class Config:
@staticmethod
def init_app(app):
pass
class LocalConfig(Config):
DEBUG = True
SQLALCHEMY_DATABASE_URI = r"sqlite:///" + os.path.join(basedir,
"data-dev.sqlite")
CELERY_BROKER_URL = 'amqp://guest:[email protected]:5672//'
config = {
"local": LocalConfig}
Приложение /exstensions.py
from flask.ext.sqlalchemy import SQLAlchemy
from celery import Celery
db = SQLAlchemy()
celery = Celery()
Приложение /factory.py
from extensions import db, celery
from flask import Flask
from flask import g
from config import config
def create_before_request(app):
def before_request():
g.db = db
return before_request
def create_app(config_name):
app = Flask(__name__)
app.config.from_object(config[config_name])
db.init_app(app)
celery.config_from_object(config)
# Register the blueprints
# Add the before request handler
app.before_request(create_before_request(app))
return app
Приложение /manage.py
from factory import create_app
app = create_app("local")
from flask import render_template
from flask import request
@app.route('/test', methods=['POST'])
def task_simple():
import tasks
tasks.do_some_stuff.delay()
return ""
if __name__ == "__main__":
app.run()
Приложение /models.py
from extensions import db
class User(db.Model):
__tablename__ = "user"
id = db.Column(db.Integer, primary_key=True)
username = db.Column(db.String(128), unique=True, nullable=False)
Приложение /tasks.py
from extensions import celery
from celery.signals import task_prerun
from flask import g, current_app
@task_prerun.connect
def close_session(*args, **kwargs):
with current_app.app_context():
# use g.db
print g
@celery.task()
def do_some_stuff():
with current_app.app_context():
# use g.db
print g
В папке приложения:
- запуск веб-сервера разработки с помощью:
python.exe manage.py
- запуск рабочих с помощью:
celery.exe worker -A tasks
Я получаю ошибку импорта, которая не имеет никакого смысла для меня. Должен ли я структурировать приложение по-другому? В конце я думаю, что я хочу довольно простую настройку, например, используя Flask с фабричным шаблоном, иметь возможность использовать расширение Flask-SQLAlchmey и иметь работника, которому нужен доступ к базе данных.
Любая помощь высоко ценится.
Traceback выполняется при запуске работника сельдерея.
Traceback (most recent call last):
File "[PATH]\scripts\celery-script.py", line 9, in <module>
load_entry_point('celery==3.1.13', 'console_scripts', 'celery')()
File "[PATH]\lib\site-packages\celery\__main__.py", line 30, in main
main()
File "[PATH]\lib\site-packages\celery\bin\celery.py", line 81, in main
cmd.execute_from_commandline(argv)
File "[PATH]\lib\site-packages\celery\bin\celery.py", line 769, in execute_from_commandline
super(CeleryCommand, self).execute_from_commandline(argv)))
File "[PATH]\lib\site-packages\celery\bin\base.py", line 305, in execute_from_commandline
argv = self.setup_app_from_commandline(argv)
File "[PATH]\lib\site-packages\celery\bin\base.py", line 473, in setup_app_from_commandline
user_preload = tuple(self.app.user_options['preload'] or ())
AttributeError: 'Flask' object has no attribute 'user_options'
ОБНОВЛЕНИЕ Я изменяю код в соответствии с предложением в комментарии. Рабочий теперь запускается, но при тестировании с запросом get на http://127.0.0.1:5000/test
. Я получаю следующую трассировку:
Traceback (most recent call last):
File "[PATH]\lib\site-packages\celery\app\trace.py", line 230, in trace_task
args=args, kwargs=kwargs)
File "[PATH]\lib\site-packages\celery\utils\dispatch\signal.py", line 166, in send
response = receiver(signal=self, sender=sender, \**named)
File "[PATH]\app\stackoverflow\tasks.py", line 7, in close_session
with current_app.app_context():
File "[PATH]\lib\site-packages\werkzeug\local.py", line 338, in __getattr__
return getattr(self._get_current_object(), name)
File "[PATH]\lib\site-packages\werkzeug\local.py", line 297, in _get_current_object
return self.__local()
File "[PATH]\lib\site-packages\flask\globals.py", line 34, in _find_app
raise RuntimeError('working outside of application context')
RuntimeError: working outside of application context exc, exc_info.traceback)))
ОБНОВЛЕНИЕ На основании комментариев от Marteen, я изменил код. Текущая рабочая версия находится по адресу: https://gist.github.com/anonymous/fa47834db2f4f3b8b257. Любые дальнейшие улучшения или предложения приветствуются.