Rumah > pembangunan bahagian belakang > Tutorial Python > Mengapakah `subprocess.Popen` dengan `readline()` digantung apabila membaca daripada skrip Ruby, dan bagaimanakah perkara ini boleh dibetulkan?

Mengapakah `subprocess.Popen` dengan `readline()` digantung apabila membaca daripada skrip Ruby, dan bagaimanakah perkara ini boleh dibetulkan?

Susan Sarandon
Lepaskan: 2024-12-19 05:44:08
asal
351 orang telah melayarinya

Why does `subprocess.Popen` with `readline()` hang when reading from a Ruby script, and how can this be fixed?

Python subprocess readlines() hang

Masalah:

Apabila membaca output skrip ruby ​​menggunakan subprocess.Popen dan readline() dalam fesyen penstriman, readline() digantung selama-lamanya dan tidak pernah kembali.

Latar Belakang:

Matlamatnya adalah untuk menstrim output fail ruby ​​baris demi baris, mencetaknya tanpa menimbal keseluruhan output.

from subprocess import Popen, PIPE, STDOUT
import pty
import os

file_path = '/Users/luciano/Desktop/ruby_sleep.rb'

command = ' '.join(["ruby", file_path])

master, slave = pty.openpty()
proc = Popen(command, bufsize=0, shell=True, stdout=slave, stderr=slave, close_fds=True)     
stdout = os.fdopen(master, 'r', 0)

while proc.poll() is None:
    data = stdout.readline()
    if data != "":
        print(data)
    else:
        break

print("This is never reached!")
Salin selepas log masuk

Skrip ruby_sleep.rb mengeluarkan mesej ringkas dengan 2 saat kelewatan:

puts "hello"

sleep 2

puts "goodbye!"
Salin selepas log masuk

Punca Punca:

readline() kekal digantung kerana skrip ruby ​​mengeluarkan data tanpa menamatkan baris (iaitu, tanpa baris baharu). Ini menyebabkan readline() menunggu selama-lamanya untuk baris baharu melengkapkan baris.

Penyelesaian:

Beberapa penyelesaian wujud bergantung pada ketersediaan platform:

  • Untuk Linux:

    Gunakan pty daripada pustaka standard untuk membuka pseudo-terminal (tty) dan mendayakan penimbal baris pada bahagian ruby, memastikan setiap baris ditamatkan dengan baris baharu.

    import os
    import pty
    from subprocess import Popen, STDOUT
    
    master_fd, slave_fd = pty.openpty()  # provide tty to enable
                                         # line-buffering on ruby's side
    proc = Popen(['ruby', 'ruby_sleep.rb'],
                 stdin=slave_fd, stdout=slave_fd, stderr=STDOUT, close_fds=True)
    os.close(slave_fd)
    try:
        while 1:
            try:
                data = os.read(master_fd, 512)
            except OSError as e:
                if e.errno != errno.EIO:
                    raise
                break # EIO means EOF on some systems
            else:
                if not data: # EOF
                    break
                print('got ' + repr(data))
    finally:
        os.close(master_fd)
        if proc.poll() is None:
            proc.kill()
        proc.wait()
    print("This is reached!")
    Salin selepas log masuk
  • Untuk platform berasaskan Linux:

    Gunakan pty dari perpustakaan standard dan pilih untuk memantau deskriptor fail induk untuk aktiviti, memastikan data dibaca dalam cara yang tidak menyekat.

    import os
    import pty
    import select
    from subprocess import Popen, STDOUT
    
    master_fd, slave_fd = pty.openpty()  # provide tty to enable
                                         # line-buffering on ruby's side
    proc = Popen(['ruby', 'ruby_sleep.rb'],
                 stdout=slave_fd, stderr=STDOUT, close_fds=True)
    
    timeout = .04 # seconds
    while 1:
        ready, _, _ = select.select([master_fd], [], [], timeout)
        if ready:
            data = os.read(master_fd, 512)
            if not data:
                break
            print("got " + repr(data))
        elif proc.poll() is not None: # select timeout
            assert not select.select([master_fd], [], [], 0)[0] # detect race condition
            break # proc exited
    os.close(slave_fd) # can't do it sooner: it leads to errno.EIO error
    os.close(master_fd)
    proc.wait()
    
    print("This is reached!")
    Salin selepas log masuk
  • Pilihan merentas platform:

    Gunakan stdbuf untuk dayakan penimbalan talian dalam bukan interaktif mod.

    from subprocess import Popen, PIPE, STDOUT
    
    proc = Popen(['stdbuf', '-oL', 'ruby', 'ruby_sleep.rb'],
                 bufsize=1, stdout=PIPE, stderr=STDOUT, close_fds=True)
    for line in iter(proc.stdout.readline, b''):
        print line,
    proc.stdout.close()
    proc.wait()
    Salin selepas log masuk

Penyelesaian ini kesemuanya mendayakan penimbalan baris pada sisi ruby, memastikan setiap baris ditamatkan dengan baris baharu, membenarkan readline() berfungsi dengan betul.

Atas ialah kandungan terperinci Mengapakah `subprocess.Popen` dengan `readline()` digantung apabila membaca daripada skrip Ruby, dan bagaimanakah perkara ini boleh dibetulkan?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

sumber:php.cn
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Artikel terbaru oleh pengarang
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan