[求助] subprocess 的 stdout 堵塞问题
要求是这样的:
1 ) 登入 mysql 服务器( mysql -h localhost -uroot -p1234 )。
2 ) 输入 mysql 内部命令 show databases,如果返回的内容出现 mysql (存在 mysql DB )就立刻强制退出整个 python 程序。
(关于要求 2 的解释,假如 show databases 的返回内容是 information_schema \r\n mysql \r\n test。不要等到 test 出现,马上就退出或者杀死该程序)
我认为只有用 subprocess 能够比较好的完成以上功能,所以以下都是以使用 subprocess 为前提。
个人试了好多方法,都不成功。
方法 1:把 stdout 放到线程里
def stdout_theard(p_stdout):
time.sleep(0.01)
for i in range(3000):
print p_stdout.readline()
s_command = 'mysql -h localhost -uroot -p1234'
sub_process = subprocess.Popen(command , stdin = subprocess.PIPE, stdout = subprocess.PIPE, stderr = subprocess.PIPE, shell = True)
thread_read_output = threading.Thread(target=stdout_theard, args=(sub_process.stdout,))
thread_read_output.setDaemon('True')
thread_read_output.start()
sub_process.stdin.write('show databases;\r\n')
方法 2:把 stdout 重定向到文件里
s_command = 'mysql -h localhost -uroot -p1234'
f_out = tempfile.TemporaryFile(mode='w+')
f_err = tempfile.TemporaryFile(mode='w+')
sub_process = subprocess.Popen(command , stdin = subprocess.PIPE, stdout = f_out, stderr = f_err, shell = True)
或者:
os.dup2(sub_process.stdout.fileno(), f_out.fileno())
以上方法都没成功。
希望前辈高手们指点。
另,以上只是用 mysql 打了个比方,实际的环境不太好说。
必须得调用一个 exe ( cisco anyConnect ),得到返回值。但登陆这个 exe 的时候,一旦给定的用户名密码错误,它会一直用这组错误的用户名密码试,直到该用户被锁死。
退出的原因就是,在第一次用户名密码错误时,就退出,避免 exe 反复试,导致锁死。 ----------------------- 以下是精选回复-----------------------
答:没成功具体是什么问题?
答:放到线程里仍旧是阻塞。
重定向到文件里,写不进去。文件总是空,只有强制退出( Ctrl+C )后,才能写到文件里,看来也是阻塞的问题。
大牛,或者说有没有成功的经验。很可能我的写法也有问题。
答:经过测试你需要 添加一个 -n 在 mysql 的命令行里
它默认开启了缓冲
答:我用 Perl 6 来测试。。
my $p = Proc::Async.new(<mysql -n -P3306 -u ovirt -pdefault>, :w);
$p.stdout.tap(&say);
$p.stderr.tap(&say);
my $pp = $p.start;
await $p.put("show databases;\r\n");
say "WAITING OVER";
await $pp;
输出
Database
information_schema
mysql
performance_schema
答:因为 python 的 subprocess 的 stdin,stdout 有阻塞的问题,所以我不会处理。
perl 没有阻塞的问题么?这到是个好消息,我可以用 perl 试试。
答:前面多余,你用一条 mysql 命令搞好了
答:mysql -u -p -e
此帖终结
答:缓冲 IO 的问题。PIPE 和普通文件默认都是全缓冲的,缓冲区没满就不会进行实际 IO,所以读不到数据。
两个办法:
1. mysql 加参数,让它强制冲洗缓冲区
2. 使用伪终端(pty),它默认是行缓冲的
分享篇博客 深入理解子进程 : http://www.kkblog.me/notes/%E6%B7%B1%E5%85%A5%E7%90%86%E8%A7%A3%E5%AD%90%E8%BF%9B%E7%A8%8B
答:为什么没有 sub_process.p.communicate()
答:这里排版不太好,我在 OSC 写了段 https://my.oschina.net/u/3573498/blog/1524999
答:不要用 readline 老老实实 read
答:还有 不要只顾 stdout 有些软件不标准
普通错误都往 stderr 里塞
0条评论