2017-11-03 3 views
2

Ich benutze Python 3.6.2, auf Fedora 26 Workstation.asyncio: Ich kann keine neue Ereignisschleife erstellen

Im Folgenden einige Scrapbook-Code ist das mein Problem veranschaulicht:

EDIT: hinzugefügt Code Sam Hartman Vorschlag.

import asyncio, json 
from autobahn.asyncio.websocket import WebSocketClientProtocol, WebSocketClientFactory 

class MyClientProtocol(WebSocketClientProtocol): 

    def onConnect(self, response): 
     print(response.peer) 

    def onOpen(self): 
     print("open") 
     self.sendMessage(json.dumps({'command': 'subscribe', 'channel': "1010"}).encode("utf8")) 

    def onMessage(self, payload, isBinary): 
     print("message") 
     print(json.loads(payload)) 

factory1 = WebSocketClientFactory("wss://api2.poloniex.com:443") 
factory1.protocol = MyClientProtocol 
loop1 = asyncio.get_event_loop() 
loop1.run_until_complete(loop1.create_connection(factory1, "api2.poloniex.com", 443, ssl=True)) 

try: 
    loop1.run_forever() 
except KeyboardInterrupt: 
    pass 
loop1.close() 

asyncio.set_event_loop(asyncio.new_event_loop()) 

factory2 = WebSocketClientFactory("wss://api2.poloniex.com:443") 
factory2.protocol = MyClientProtocol 
loop2 = asyncio.get_event_loop() 
loop2.run_until_complete(loop2.create_connection(factory2, "api2.poloniex.com", 443, ssl=True)) 

try: 
    loop2.run_forever() 
except KeyboardInterrupt: 
    pass 
loop2.close() 

Nachdem geschlossen eine anfängliche asyncio Ereignisschleife, eine andere zu schaffen und sie als die globale Ereignisschleife Einstellung versuchen, die neue Ereignisschleife zu verwenden, ergibt sich folgende Fehler:

Fatal write error on socket transport 
protocol: <asyncio.sslproto.SSLProtocol object at 0x7f8a84ed4748> 
transport: <_SelectorSocketTransport fd=6> 
Traceback (most recent call last): 
    File "/usr/lib64/python3.6/asyncio/selector_events.py", line 762, in write 
    n = self._sock.send(data) 
OSError: [Errno 9] Bad file descriptor 
Fatal error on SSL transport 
protocol: <asyncio.sslproto.SSLProtocol object at 0x7f8a84ed4748> 
transport: <_SelectorSocketTransport closing fd=6> 
Traceback (most recent call last): 
    File "/usr/lib64/python3.6/asyncio/selector_events.py", line 762, in write 
    n = self._sock.send(data) 
OSError: [Errno 9] Bad file descriptor 

During handling of the above exception, another exception occurred: 

Traceback (most recent call last): 
    File "/usr/lib64/python3.6/asyncio/sslproto.py", line 648, in _process_write_backlog 
    self._transport.write(chunk) 
    File "/usr/lib64/python3.6/asyncio/selector_events.py", line 766, in write 
    self._fatal_error(exc, 'Fatal write error on socket transport') 
    File "/usr/lib64/python3.6/asyncio/selector_events.py", line 646, in _fatal_error 
    self._force_close(exc) 
    File "/usr/lib64/python3.6/asyncio/selector_events.py", line 658, in _force_close 
    self._loop.call_soon(self._call_connection_lost, exc) 
    File "/usr/lib64/python3.6/asyncio/base_events.py", line 574, in call_soon 
    self._check_closed() 
    File "/usr/lib64/python3.6/asyncio/base_events.py", line 357, in _check_closed 
    raise RuntimeError('Event loop is closed') 
RuntimeError: Event loop is closed 
Exception in callback _SelectorSocketTransport._read_ready() 
handle: <Handle _SelectorSocketTransport._read_ready()> 
Traceback (most recent call last): 
    File "/usr/lib/python3.6/site-packages/txaio/_common.py", line 63, in call_later 
    self._buckets[real_time][1].append(call) 
KeyError: 412835000 

During handling of the above exception, another exception occurred: 

Traceback (most recent call last): 
    File "/usr/lib64/python3.6/asyncio/events.py", line 127, in _run 
    self._callback(*self._args) 
    File "/usr/lib64/python3.6/asyncio/selector_events.py", line 731, in _read_ready 
    self._protocol.data_received(data) 
    File "/usr/lib64/python3.6/asyncio/sslproto.py", line 503, in data_received 
    ssldata, appdata = self._sslpipe.feed_ssldata(data) 
    File "/usr/lib64/python3.6/asyncio/sslproto.py", line 204, in feed_ssldata 
    self._handshake_cb(None) 
    File "/usr/lib64/python3.6/asyncio/sslproto.py", line 619, in _on_handshake_complete 
    self._app_protocol.connection_made(self._app_transport) 
    File "/usr/lib/python3.6/site-packages/autobahn/asyncio/websocket.py", line 97, in connection_made 
    self._connectionMade() 
    File "/usr/lib/python3.6/site-packages/autobahn/websocket/protocol.py", line 3340, in _connectionMade 
    WebSocketProtocol._connectionMade(self) 
    File "/usr/lib/python3.6/site-packages/autobahn/websocket/protocol.py", line 1055, in _connectionMade 
    self.onOpenHandshakeTimeout, 
    File "/usr/lib/python3.6/site-packages/txaio/_common.py", line 72, in call_later 
    self._notify_bucket, real_time, 
    File "/usr/lib/python3.6/site-packages/txaio/aio.py", line 382, in call_later 
    return self._config.loop.call_later(delay, real_call) 
    File "/usr/lib64/python3.6/asyncio/base_events.py", line 543, in call_later 
    timer = self.call_at(self.time() + delay, callback, *args) 
    File "/usr/lib64/python3.6/asyncio/base_events.py", line 553, in call_at 
    self._check_closed() 
    File "/usr/lib64/python3.6/asyncio/base_events.py", line 357, in _check_closed 
    raise RuntimeError('Event loop is closed') 
RuntimeError: Event loop is closed 

Es scheint vernünftig, dass Eventuell müssen Sie eine Ereignisschleife erneut öffnen, nachdem Sie eine frühere geschlossen haben. Tatsächlich diese Frage auch zeigt, wie: Asyncio Event Loop is Closed

Der folgende Code sollte dies erreichen:

loop = asyncio.new_event_loop() 
asyncio.set_event_loop(loop) 

so bin ich eindeutig etwas falsch zu machen. Kann jemand etwas vermisst sehen?

+0

Haben Sie versucht, läuft 'loop.stop()' statt 'close()' zu rekonstruieren? –

+0

Die Ereignisschleife läuft nicht an dem Punkt, an dem ich 'loop.close()' rufe, also wäre es ein No-Op. –

Antwort

1

Ich habe recht hohe Zuversicht, dass Ihr Factory-Objekt einen Verweis auf die alte Ereignisschleife behält, vermutlich von asyncio.get_event_loop. Asyncio-Konsumenten sind schlecht darin, versteckte Verweise auf Schleifen zu bekommen.

Meine Empfehlung ist es, die Web-Socket-Factory nach dem Schließen der Schleife

+0

Danke Sam. Ich habe versucht, die Factory zu rekonstruieren, sogar mit verschiedenen Variablennamen für die erste und zweite Factory/Loop, aber ich bekomme immer noch die gleichen Fehler ('OSError: [Errno 9] Fehlerhafter Dateideskriptor',' RuntimeError: Ereignisschleife ist geschlossen', 'KeyError: 15068000'). Es ist, als ob der Aufruf 'set_event_loop' einfach nicht funktioniert. Denkst du, ich sollte das als Fehler melden? –

+1

Sie können prüfen, ob set_event_loop mit einem Assert arbeitet, nachdem get_event_loop den Wert zurückgibt, auf den Sie es gesetzt haben. Es ist fast sicher baumelnden Referenzen irgendwo in Ihrem Websocket-Implementierung –

+0

OK. Ich werde die Autobahn.asyncio Leute fragen. Danke Sam. –

Verwandte Themen