Created
January 20, 2017 07:39
-
-
Save blubberdiblub/164d4b418d3b0ca0da3006f48dcfbbbc to your computer and use it in GitHub Desktop.
Minimal console output of a subprocess with Tk.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#!/usr/bin/env python3 | |
# -*- coding: utf-8 -*- | |
from subprocess import Popen, PIPE, STDOUT, TimeoutExpired | |
from threading import Thread, Event | |
from queue import Queue, Empty | |
from tkinter import Tk, Text, END | |
class ProcessOutputReader(Thread): | |
def __init__(self, queue, cmd, params=(), | |
group=None, name=None, daemon=True): | |
super().__init__(group=group, name=name, daemon=daemon) | |
self._stop_request = Event() | |
self.queue = queue | |
self.process = Popen((cmd,) + tuple(params), | |
stdout=PIPE, | |
stderr=STDOUT, | |
universal_newlines=True) | |
def run(self): | |
for line in self.process.stdout: | |
if self._stop_request.is_set(): | |
# if stopping was requested, terminate the process and bail out | |
self.process.terminate() | |
break | |
self.queue.put(line) # enqueue the line for further processing | |
try: | |
# give process a chance to exit gracefully | |
self.process.wait(timeout=3) | |
except TimeoutExpired: | |
# otherwise try to terminate it forcefully | |
self.process.kill() | |
def stop(self): | |
# request the thread to exit gracefully during its next loop iteration | |
self._stop_request.set() | |
# empty the queue, so the thread will be woken up | |
# if it is blocking on a full queue | |
while True: | |
try: | |
self.queue.get(block=False) | |
except Empty: | |
break | |
self.queue.task_done() # acknowledge line has been processed | |
class MyConsole(Text): | |
def __init__(self, parent, queue, update_interval=50, process_lines=500): | |
super().__init__(parent) | |
self.queue = queue | |
self.update_interval = update_interval | |
self.process_lines = process_lines | |
self.after(self.update_interval, self.fetch_lines) | |
def fetch_lines(self): | |
something_inserted = False | |
for _ in range(self.process_lines): | |
try: | |
line = self.queue.get(block=False) | |
except Empty: | |
break | |
self.insert(END, line) | |
self.queue.task_done() # acknowledge line has been processed | |
# ensure scrolling the view is at most done once per interval | |
something_inserted = True | |
if something_inserted: | |
self.see(END) | |
self.after(self.update_interval, self.fetch_lines) | |
# create the root widget | |
root = Tk() | |
# create a queue for sending the lines from the process output reader thread | |
# to the TkInter main thread | |
line_queue = Queue(maxsize=1000) | |
# create a process output reader | |
reader = ProcessOutputReader(line_queue, 'python3', params=['-u', 'test.py']) | |
# create a console | |
console = MyConsole(root, line_queue) | |
reader.start() # start the process | |
console.pack() # make the console visible | |
root.mainloop() # run the TkInter main loop | |
reader.stop() | |
reader.join(timeout=5) # give thread a chance to exit gracefully | |
if reader.is_alive(): | |
raise RuntimeError("process output reader failed to stop") |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment