概要
- Python3では、asyncioモジュールを利用することでUNIXドメインソケットによるプロセス間通信(IPC)が可能である。
- クライアントプロセスからソケットに書き込まれたバイト列を読み込む際に、書き込みが終わっているにもかかわらずサーバープロセス側の読み込みが終わらないという事態が発生した。
-
書き込み終了時に
write_eof()
を呼び出すことでサーバープロセス側の読み込みが終了した。
環境
- Windows Subsystem for Linuxを使用
- ホストOS: Windows 10 Home 64bit
- ゲストOS: Ubuntu 18.04.3 LTS
- Python 3.8.0
#事例
Responderで作成するサイトと自作の計算プログラムのプロセス間通信にUNIXドメインソケットを使おうと考え、前段階としてソケットの挙動を確認するスクリプトを書いていました。
テストの処理の流れは
-
asyncio.start_unix_server(コールバック, ソケットのパス)
でソケットを作成し、サーバーを起動する - クライアントは
asyncio.open_unix_connection(ソケットのパス)
でサーバーと接続する - クライアントがソケットに小文字の文字列を書き込む
- サーバーがソケットから小文字の文字列を読み込む
- 読み込んだ文字列を大文字に変換する
- 変換した文字列をソケットに書き込む
- クライアントがソケットから大文字の文字列を読み込んで表示する
としました。
クライアントサイド:
import asyncio
import datetime
def now():
return "[{}]".format(datetime.datetime.now().time())
async def client(message):
"socketサーバーに文字列を送って応答を読みだす"
# コネクションの作成
reader, writer = await asyncio.open_unix_connection(path="./sock")
#メッセージの送信
writer.write(message.encode())
print(now(), "send {}".format(message)
await writer.drain()
# メッセージの受信
res = await reader.read(-1)
print(now(), "receive {}".format(res))
# 処理の終了
writer.close()
await writer.wait_closed()
if __name__ == "__main__":
asyncio.run(client("hogehoge"))
サーバーサイド:
import asyncio
import datetime
import os
def now():
return "[{}]".format(datetime.datetime.now().time())
async def handler(reader, writer):
"""
ドメインソケットに接続があった時のコールバック関数
受信した文字列を大文字にして返す
"""
print(now(), "handler called.")
# メッセージの受信
data = await reader.read(-1) # 引数を-1にするとEOFまで読む
print(now(), "recieve {}".format(data))
# メッセージの処理と返信
res = str(data.decode()).upper()
writer.write(res.encode())
await writer.drain()
print(now(), "write complete.")
#終了処理
writer.close()
await writer.wait_closed()
async def main():
#ソケットの作成と読み書き権限の付与
server = await asyncio.start_unix_server(handler, path="./sock")
os.chmod("./sock", 0o600)
#サーバーの起動
async with server:
await server.serve_forever()
if __name__ == "__main__":
asyncio.run(main())
実行結果:
user@DESKTOP-XXX:~$ python socket_client_badcase.py
[11:39:59.574727] send hogehoge
user@DESKTOP-XXX:~$ python socket_server.py
[11:39:59.574847] handler called.
このように、ソケットへのメッセージの書き込みは終了し、メッセージハンドラも呼び出されているものの、サーバーサイドでのソケットからの読み込みが終わらず処理がストップしてしまいました。
解決方法
公式リファレンスのStreamsのページを読み返すと、
class asyncio.StreamWriter
(中略)
write_eof()
Close the write end of the stream after the buffered write data is flushed.
(https://docs.python.org/ja/3/library/asyncio-stream.html より引用)
という記述がありました。書き込み終了時にこれを呼び出し、読み込んでいる側にファイルの終わりを伝える必要があるようです。
これを踏まえてクライアントサイドのスクリプトを書き直すと:
import asyncio
import datetime
def now():
return "[{}]".format(datetime.datetime.now().time())
async def client(message):
"socketサーバーに文字列を送って応答を読みだす"
# コネクションの作成
reader, writer = await asyncio.open_unix_connection(path="./sock")
#メッセージの送信
writer.write(message.encode())
print(now(), "send {}".format(message))
writer.write_eof() # ここを追加
await writer.drain()
# メッセージの受信
res = await reader.read(-1) # 引数を-1にするとEOFまで読む
print(now(), "receive {}".format(res))
# 処理の終了
writer.close()
await writer.wait_closed()
if __name__ == "__main__":
asyncio.run(client("hogehoge"))
実行結果:
user@DESKTOP-XXX:~$ python socket_client.py
[12:14:17.147107] send hogehoge
[12:14:17.153610] receive b'HOGEHOGE'
user@DESKTOP-XXX:~$ python socket_server.py
[12:14:17.147750] handler called.
[12:14:17.150608] recieve b'hogehoge'
[12:14:17.151640] write complete.
このようにクライアントとサーバーの間でデータのやり取りが完了しました。
#まとめ
Pythonのasyncioモジュールを利用したUNIXドメインソケットでは、書き込み終了を読み込んでいるプロセスに伝えるためにwrite_eof()
を呼び出す必要があります。これにより、読み出し側に書き込みの終了が通知されるようです。
#後書き
初投稿です。脱ROM専