微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

线程“Thread-1”中没有当前事件循环

如何解决线程“Thread-1”中没有当前事件循环

我对 python 非常陌生,我正在尝试实现一个 API,用于从盈透证券交易平台请求一些数据。我正在使用 Flask 和 ib_insync

这是我的代码

import configparser
import json
import logging.config

from flask import Flask
from ib_insync import IB

flask = Flask(__name__)

ib = IB()

config = configparser.ConfigParser()
config.read('settings.ini')

logging.config.fileConfig('logging.conf')
logger = logging.getLogger('ib-custom')


@flask.route('/health',methods=['GET'])
def health():
    return 'OK'


@flask.route('/summary',methods=['GET'])
def account_summary():
    logger.debug('account_summary called')
    summary = ib.accountSummary()
    return json.dumps([ob._asdict() for ob in summary])


if __name__ == '__main__':
    ib.connect('127.0.0.1',7497,clientId=1)
    logger.info('Application started')
    flask.run(debug=True,use_reloader=False)

当我尝试执行 localhost:5000/summary 时会引发错误。这是堆栈跟踪:

Traceback (most recent call last):
  File "/Users/irusanov/Projects/PycharmProjects/ib-custom/venv/lib/python3.8/site-packages/flask/app.py",line 2464,in __call__
    return self.wsgi_app(environ,start_response)
  File "/Users/irusanov/Projects/PycharmProjects/ib-custom/venv/lib/python3.8/site-packages/flask/app.py",line 2450,in wsgi_app
    response = self.handle_exception(e)
  File "/Users/irusanov/Projects/PycharmProjects/ib-custom/venv/lib/python3.8/site-packages/flask/app.py",line 1867,in handle_exception
    reraise(exc_type,exc_value,tb)
  File "/Users/irusanov/Projects/PycharmProjects/ib-custom/venv/lib/python3.8/site-packages/flask/_compat.py",line 39,in reraise
    raise value
  File "/Users/irusanov/Projects/PycharmProjects/ib-custom/venv/lib/python3.8/site-packages/flask/app.py",line 2447,in wsgi_app
    response = self.full_dispatch_request()
  File "/Users/irusanov/Projects/PycharmProjects/ib-custom/venv/lib/python3.8/site-packages/flask/app.py",line 1952,in full_dispatch_request
    rv = self.handle_user_exception(e)
  File "/Users/irusanov/Projects/PycharmProjects/ib-custom/venv/lib/python3.8/site-packages/flask/app.py",line 1821,in handle_user_exception
    reraise(exc_type,line 1950,in full_dispatch_request
    rv = self.dispatch_request()
  File "/Users/irusanov/Projects/PycharmProjects/ib-custom/venv/lib/python3.8/site-packages/flask/app.py",line 1936,in dispatch_request
    return self.view_functions[rule.endpoint](**req.view_args)
  File "/Users/irusanov/Projects/PycharmProjects/ib-custom/main.py",line 27,in account_summary
    summary = ib.accountSummary()
  File "/Users/irusanov/Projects/PycharmProjects/ib-custom/venv/lib/python3.8/site-packages/ib_insync/ib.py",line 405,in accountSummary
    return self._run(self.accountSummaryAsync(account))
  File "/Users/irusanov/Projects/PycharmProjects/ib-custom/venv/lib/python3.8/site-packages/ib_insync/ib.py",line 310,in _run
    return util.run(*awaitables,timeout=self.RequestTimeout)
  File "/Users/irusanov/Projects/PycharmProjects/ib-custom/venv/lib/python3.8/site-packages/ib_insync/util.py",line 291,in run
    loop = asyncio.get_event_loop()
  File "/Library/Developer/CommandLinetools/Library/Frameworks/python3.framework/Versions/3.8/lib/python3.8/asyncio/events.py",line 639,in get_event_loop
    raise RuntimeError('There is no current event loop in thread %r.'
RuntimeError: There is no current event loop in thread 'Thread-1'.

为什么会出现这个错误?如何解决

解决方法

似乎 Flask 使用其自己的内部阻塞/回调/线程模型,并且与构建此项目的 Asyncio 不兼容。来源:https://github.com/erdewit/ib_insync/issues/266

可以使用另一个框架而不是 Flask 来实现所需的行为。例如。夸脱。

在使用 Quart 的情况下,可以重写此代码:

import asyncio
import configparser
import json
import logging.config
import nest_asyncio

from quart import Quart
from ib_insync import IB,util

nest_asyncio.apply()
qrt = Quart(__name__)

ib = IB()

config = configparser.ConfigParser()
config.read('settings.ini')

logging.config.fileConfig('logging.conf')
logger = logging.getLogger('ib-custom')


@qrt.route('/health',methods=['GET'])
def health():
    return 'OK'


@qrt.route('/summary')
async def summary():
    with await IB().connectAsync() as ibi:
        acct = ibi.managedAccounts()[0]
        summary = ibi.accountSummary(acct)
        await ibi.accountSummaryEvent
        resp = json.dumps(util.tree(summary))
    return resp


@qrt.route('/pnl')
async def pnl():
    with await IB().connectAsync() as ibi:
        acct = ibi.managedAccounts()[0]
        pnl = ibi.reqPnL(acct)
        await ibi.pnlEvent
        resp = json.dumps(util.tree(pnl))
    return resp


if __name__ == '__main__':
    # ib.connect('127.0.0.1',7497,clientId=1)
    logger.info('Application started')
    qrt.run(debug=True)

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。