次の単純なスクリプトはサブプロセスでハングします。断続的に呼び出します(約30%)。
use_lock = Trueでない限り、それがハングしないので、サブプロセスがスレッドセーフではないと私を導いてくれます! 予想される動作は、スクリプトが5〜6秒以内に終了することです。
バグを示すには、 "python bugProof.py"を何回か実行してハングアップしてください。 Ctrl-Cが終了します。ポポポンは1回か2回しか現れませんが、3回目は現れません。subprocess.Popenはスレッドセーフではありませんか?
#!/bin/sh
echo "Calculating factorial (anything that's somewhat compute intensive, this script takes 3 sec on my machine"
ans=1
counter=0
fact=999
while [ $fact -ne $counter ]
do
counter=`expr $counter + 1`
ans=`expr $ans \* $counter`
done
echo "Factorial calculation done"
read -p "Test input (this part is critical for bug to occur): " buf
echo "$buf"
システム情報: Linuxの2.6.32-358.123.2.openstack.el6.x86_64#1 SMP木9月(factorial.sh)上で参照
import subprocess, threading, fcntl, os, time
end_time = time.time()+5
lock = threading.Lock()
use_lock = False
path_to_factorial = os.path.join(os.path.dirname(os.path.realpath(__file__)),'factorial.sh')
def testFunction():
print threading.current_thread().name, '| pre-Popen'
if use_lock: lock.acquire()
p = subprocess.Popen([path_to_factorial], stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
if use_lock: lock.release()
print threading.current_thread().name, '| post-Popen'
fcntl.fcntl(p.stdout, fcntl.F_SETFL, os.O_NONBLOCK)
fcntl.fcntl(p.stderr, fcntl.F_SETFL, os.O_NONBLOCK)
while time.time()<end_time:
try: p.stdout.read()
except: pass
try: p.stderr.read()
except: pass
print threading.current_thread().name, '| DONE'
for i in range(3):
threading.Thread(target=testFunction).start()
シェルスクリプト26 17:14:58 EDT 2013 x86_64 x86_64 x86_64 GNU/Linux
Python 2.7.3(デフォルト、Jan 22 2013、11:34:30)
[GCC 4.4.6 20120305(Red Hat 4.4.6-4) ] on linux2
Popenコールでclose_fds = Trueを設定すると、何らかの理由で問題が解決されます。 – Roman
これは文字通り、私が最近Pythonで遭遇した最も怒っていたことでした。すごくイライラしてる。 –