【问题标题】:Flask threads share valuesFlask 线程共享值
【发布时间】:2017-12-30 09:14:58
【问题描述】:

我有一个 Flask 应用程序,它可以提取数据,然后转换该数据,然后将这些结果上传到数据仓库。我想将这一系列步骤“记录”到 Web UI,如果需要,还记录警告。

在同一个 Flask 应用程序中,我实现了 SSE(服务器发送事件)并且它可以工作,但我无法“更新”事件以反映应用程序的状态。有人可以告诉我如何在不断更新浏览器的 SSE 和正在运行的应用程序之间“共享”数据,以便让用户了解数据处理的最新情况吗?或者也许是不同的方法?欢迎所有想法。我是 Flask 的新手。

这是 UI 所在的 views.py 模块:

from flask import render_template
import logging

logger = logging.getLogger(__name__)


def index():
    """ User Interface page """
    return render_template('index.html')

index.html

<!DOCTYPE html>
<html>
    <head>
        <title>Status</title>
        <script type="text/javascript" src="/static/js/sse.js"></script>
    </head>
    <body>
        <h1>Status</h1>
        <div id="status"></div>
    </body>
</html>

sse.js 在这里:

var sse = new EventSource('/status');
sse.onmessage = function(e) {
    console.log(e.data);
    document.getElementById('status').innerHTML = e.data;
};

这是 SSE 代码本身:

class ServerSentEvents(object):

    def __init__(self, data):
        self.data = data
        self.event = None
        self.id = None
        self.desc_map = {
            'data': self.data,
            'event': self.event,
            'id': self.id}

    def encode(self):
        if not self.data:
            return ''

        lines = ['{}: {}'.format(k, v) for k, v in self.desc_map.items() if v]
        return '{}\n\n'.format('\n'.join(lines))

Redis 的状态实现:

from flask import Response
from .sse import ServerSentEvents
import logging

logger = logging.getLogger(__name__)


class EntityStatus():
    """ Broadcast application state """

    def __init__(self, red):
        self.red = red

    def status(self):

        def ready():
            pubsub = self.red.pubsub()
            pubsub.subscribe('status')
            for message in pubsub.listen():
                logger.debug('Status: {}'.format(message['data']))
                ev = ServerSentEvents(message['data'])
                yield ev.encode()

        return Response(ready(), mimetype='text/event-stream')

这里是路由模块:

from .version import __version__
from .actions import index
from .actions import EntityStatus


def add_routes(app, red, etl):

    def _add_headers(response):
        """ Add custom response headers """
        response.headers['X-App-Version'] = __version__
        response.headers['User-Agent'] = 'Segments Engine'
        return response

    app.after_request(_add_headers)

    api = EntityStatus(red)

    app.add_url_rule('/', 'index', index)
    app.add_url_rule('/status', 'status', api.status)
    app.add_url_rule('/etl', 'etl', etl.etl)

这里是发送状态的代码摘录:

class EntityApi(object):
    """ Entity API """

    def __init__(self, debug, red, data):
        self.debug = debug
        self.red = red

        self.red.publish('status', 'API Ready')

“索引”页面加载了一个js脚本,该脚本成功地指向和接收来自服务器的SSE数据并正确显示数据。 'status' 是发送数据的地方,'etl' 是进行实际数据处理的地方。

【问题讨论】:

  • etl视图函数会同时处理多个请求吗?

标签: python multithreading flask redis sse


【解决方案1】:

我修改了您现在正在查看的代码。使用 Redis 和 SSE,我现在已经设法在处理数据时将应用程序状态的更新发布到 Web UI :) 我发现的通知示例和其他相关的 Redis 和 Celery 实现帮助我做到了这一点,谢谢大家 :)我只需要一个简单的解决方案,以后可以从这里改进和扩展。

【讨论】:

    猜你喜欢
    • 2014-07-14
    • 2017-05-05
    • 1970-01-01
    • 2017-05-11
    • 1970-01-01
    • 2016-06-22
    • 1970-01-01
    • 1970-01-01
    • 2010-12-25
    相关资源
    最近更新 更多