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

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

新規登録して質問してみよう
ただいま回答率
85.46%
Ubuntu

Ubuntuは、Debian GNU/Linuxを基盤としたフリーのオペレーティングシステムです。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

Q&A

解決済

2回答

3392閲覧

一定期間subprocessの標準出力がない事を検知したい

Reem

総合スコア27

Ubuntu

Ubuntuは、Debian GNU/Linuxを基盤としたフリーのオペレーティングシステムです。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

0グッド

0クリップ

投稿2020/02/22 11:49

前提・実現したいこと

初歩的な質問かもしれませんが、自己解決出来なかったため質問させていただきます。

subprocessで起動したサーバーソフトの起動ログ(標準出力)のみを取得するためprco.stdoutをforループで回しています。

Python

1command = ['./program'] 2P_PATH = '/home/user/デスクトップ/NewFolder' 3 4prco = subprocess.Popen(command, cwd=P_PATH, stdout = subprocess.PIPE, stderr = subprocess.STDOUT) 5 6for line in prco.stdout: 7 print(line) 8 9print("finish")

(forを使うと内部でイテレータが動いているらしいです)subprocessで起動したサーバーソフトを終了しないと要素が無くなった判定にならないのかStopIteration例外が出ず、forループから抜け出すことができません。

起動ログを取得したあと別の処理に移りたいので、一定期間出力がなければループを抜けたいのですが検知する方法が分からず困っています。

Python、Linux系OS初心者なので細かいところまで解説していただけると幸いです。

ご回答よろしくお願いします。

環境

Python 3.6.9
Ubuntu 18.04.4 LTS

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

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

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

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

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

guest

回答2

0

ベストアンサー

timeout on subprocess readline in pythonの回答が参考になります。
以下では5秒出力がなければ起動したプロセスを終了し、後の処理を行うコード例です。

Python

1import asyncio 2import sys 3from asyncio.subprocess import PIPE, STDOUT 4 5# 何らかの処理。単にlineを出力 6def do_something(line): 7 print(line.decode().rstrip()) 8 return True 9 10async def run_command(*args, timeout=None): 11 # start child process 12 # NOTE: universal_newlines parameter is not supported 13 process = await asyncio.create_subprocess_exec(*args, 14 stdout=PIPE, stderr=STDOUT) 15 16 # read line (sequence of bytes ending with b'\n') asynchronously 17 while True: 18 try: 19 line = await asyncio.wait_for(process.stdout.readline(), timeout) 20 except asyncio.TimeoutError: 21 print('timeout') 22 else: 23 if not line: # EOF 24 break 25 elif do_something(line): 26 continue # while some criterium is satisfied 27 process.kill() # timeout or some criterium is not satisfied 28 break 29 return await process.wait() # wait for the child process to exit 30 31 32if __name__ == '__main__': 33 34 if sys.platform == "win32": 35 loop = asyncio.ProactorEventLoop() # for subprocess' pipes on Windows 36 asyncio.set_event_loop(loop) 37 else: 38 loop = asyncio.get_event_loop() 39 40 # 5秒出力がなければ終了する 41 args = ('python', 'test.py') 42 returncode = loop.run_until_complete(run_command(*args, timeout=5)) 43 loop.close() 44 45 # 後の処理 46 # : 47 print('after proc')

test.py

Python

1import time 2import sys 3 4for i in range(10): 5 sys.stdout.write('{}\n'.format(i)) 6 sys.stdout.flush() 7 time.sleep(i+1)

実行結果

0 1 2 3 4 5 timeout after proc

投稿2020/02/22 13:53

can110

総合スコア38278

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

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

Reem

2020/02/23 06:07

ご回答ありがとうございます。 asyncio.create_subprocess_execとasyncio.wait_forで実現したい機能を作ることができました! とても勉強になりました。
guest

0

subprocessで実行しているコマンドが終了したかどうかを検出できればいいのでは

投稿2020/02/22 11:52

y_waiwai

総合スコア87800

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

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

Reem

2020/02/22 12:27

ご回答ありがとうございます。 分かりづい質問をしてしまいすみません。 subprocessで実行しているのはサーバーソフトなので、コマンドが終了したのを検知する方法だとサーバーソフトが終了した時にしかログが取得出来ません。 意図した回答と少し違うので、よろしければ再度ご回答お願いします。
y_waiwai

2020/02/22 12:35

表題の通りなら、標準出力がなされた時点の時刻を取得しておいて、現在時刻とある程度の時間経過があったかどうかを判定すればいいという話となります
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.46%

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

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

質問する

関連した質問