質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.35%
Python 3.x

Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

WebSocket

WebSocketとは双方向・全二重コミュニケーションのためのAPIでありプロトコルのことを指します。WebSocketはHTML5に密接に結びついており、多くのウェブブラウザの最新版に導入されています。

Q&A

解決済

2回答

3805閲覧

websocketsで取得したメッセージをforで表示したい

shirasublue

総合スコア19

Python 3.x

Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

WebSocket

WebSocketとは双方向・全二重コミュニケーションのためのAPIでありプロトコルのことを指します。WebSocketはHTML5に密接に結びついており、多くのウェブブラウザの最新版に導入されています。

0グッド

0クリップ

投稿2020/10/06 02:37

編集2020/10/12 23:45

実行環境

  • Python 3.8.2
  • websockets 8.1

実現したいこと

websocketsライブラリにおいて、asyncioを隠蔽したいです。

以下コードの for で~~'piyo'~~受信メッセージを表示したいです。
実際にはwebsocketでの受信を逐次表示するような用途で使用したいです。
hogera()より上は、実際にはライブラリ内に入るイメージです。

  • 記載変更1 websockets を利用する場合、最初のコードでは全く意味がないと思われたので現時点でのコードを追加しました。

  • 記載変更2 最低限守りたいコードが以下です。

for msg in Hoge: print(msg)
  • 記載変更3 実際には自作のライブラリに入ります。

ライブラリ使用者からasyncioを隠蔽するのが目的です。

現状参考コード

python3

1import websockets 2import asyncio 3import json 4 5class Hoge(object): 6 async def stream(self): 7 async with websockets.connect(self.uri) as ws: 8 while not ws.closed: 9 response = await ws.recv() 10 self.message = json.loads(response) 11 12 def __init__(self): 13 self.message = None 14 15 def __iter__(self): 16 return self 17 18 def __next__(self): 19 return self.message 20 21''' いずこへ… 22 loop = asyncio.get_event_loop() 23 loop.create_task(self.stream()) 24 loop.run_forever() 25''' 26 27for msg in Hoge: 28 print(msg)

最初に投稿したコード

Python3

1import asyncio 2 3async def hoge(): 4 while(True): 5 fuga = 'piyo' 6 7def hogera(): 8 loop = asyncio.get_event_loop() 9 loop.create_task(hoge()) 10 loop.run_forever 11 12msgs = hogera() 13for msg in msgs: 14 print(msg) #ここでpiyopiyo言ってほしい

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

toast-uz

2020/10/12 11:49

現在の実行結果を記載下さい
shirasublue

2020/10/12 23:41

現在のコードは参考としての掲載で、動作を意図していません。 Traceback (most recent call last): File "teratail.py", line 29, in <module> for msg in Hoge: TypeError: 'type' object is not iterable
toast-uz

2020/10/12 23:52 編集

期待される動作としては、「websocketsに接続しっぱなしにして、非同期に受信するメッセージ群を、for文で逐次表示させたい」ということでしょうか?
guest

回答2

0

このようなコードで質問者様の意図は実現できるようです。
「非同期に受信するメッセージ群を、for文で処理する」には、for文そのものを非同期にする必要があります。

websocketsライブラリでコネクションを確立したまま持ち回るコードサンプルがあまりなく、苦労しました。
以下のサンプルを見ながら作りました。
websocket_connector.py
なお、json変換はしていません。

Python

1import websockets 2import asyncio 3 4class Hoge: 5 async def connect(self, uri): 6 self.ws : websockets.WebSocketCommonProtocol = await websockets.connect(uri) 7 8 def __aiter__(self): 9 return self 10 11 async def __anext__(self): 12 response = await self.ws.recv() 13 return response 14 15async def main(): 16 hoge = Hoge() 17 await hoge.connect('ws://localhost:8000/') 18 async for msg in hoge: 19 print(msg) 20 21asyncio.run(main())

投稿2020/10/13 11:06

編集2020/10/13 11:12
toast-uz

総合スコア3266

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

shirasublue

2020/10/13 22:05

ありがとうございます。 目的としてはクラス使用者からのasyncioの隠蔽ですが、貴殿のコードでもスマートな書き方が出来そうなので、もし不可能そうであれば貴殿の方法の使用を検討します。
toast-uz

2020/10/13 22:58

同期処理である普通のforループで、非同期で散発的に発生するメッセージを渡すことは、原理的にできません。その時点で到着済のメッセージだけを吐き出すようなforループ(多くの場合は何も吐き出されない)なら作成可能です。利用者からはもう1つ外にwhile Trueの無限ループを作って、sleepを入れてぐるぐるまわす形になります。そういうものをお望みでしょうか?
toast-uz

2020/10/13 23:01 編集

その場合、class全体をイテレーションするような現状の書き方ではなく、到着済メッセージを管理するqueueだけを利用側に見せる実装になると思います。当初に確認していただいた「websocketsに接続しっぱなしにして、非同期に受信するメッセージ群を、for文で逐次表示させたい」からは実装方針が大きく離れると思います。
shirasublue

2020/10/14 03:20

仰るとおり、ホットループが発生するような実装で好ましくないと判断したため、デコレータによる実装へと方針を変えました。ご検討ありがとうございました。
toast-uz

2020/10/14 03:24

asyncioを使う以上は非同期処理が露出してしまうのをどう解決するか悩ましかったのですが、デコレータという解決策は参考になります。こちらこそありがとうごさいます。
guest

0

自己解決

質問主題から離れてしまいましたが、asyncio の隠蔽のためデコレータによる flask のような形のコードとしました。
(以下サンプルコードは1ファイルのため asyncio が入っていますが、自作ライブラリに搭載する形で問題なく動作しました。)

Python3

1import websockets 2import asyncio 3 4class EntitySpec(object): 5 def __init__(self, ctx): 6 self.ctx = ctx 7 self.uri = "ws://{}:{}".format( 8 ctx.hostname, ctx.port, 9 ) 10 11 def __call__(self, func): 12 async def stream(func): 13 async with websockets.connect(self.uri) as ws: 14 while not ws.closed: 15 try: 16 response = await ws.recv() 17 func(response) 18 except websockets.exceptions.ConnectionClosedOK: 19 self.loop.stop() 20 self.loop = asyncio.get_event_loop() 21 self.loop.create_task(stream(func)) 22 return stream 23 24 def run(self): 25 self.loop.run_forever() 26 27class Context(object): 28 def __init__( 29 self, 30 hostname='localhost', 31 port=8765, 32 ): 33 self.hostname = hostname 34 self.port = port 35 self.websocket = EntitySpec(self) 36 37api = Context() 38 39@api.websocket 40def reciever(msg): 41 print(msg) 42 43api.websocket.run() 44

投稿2020/10/14 03:17

shirasublue

総合スコア19

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.35%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問