Python使用subprocess.Popen

时间:2016-03-14 13:30:49

标签: python django multithreading

我在一个简单的gui应用程序上工作。

在我的实际案例中,这个应用程序启动Django( python manage.py runserver )webserver并关闭Django webserver。为了简化我的应用程序的可测试性,我通过 sleep 命令替换django start webserver。

当我触发启动服务器按钮时:服务器启动。 然后,我触发停止服务器按钮,然后关闭该过程。

但是django webserver(或sleep命令)仍在工作。

如何停止 subprocess.Popen 调用的进程并停止de命令?

有我的代码:

#!/usr/bin/python
# coding:utf-8
import os
import sys
import subprocess
import time
import datetime
import zipfile
import subprocess
from concurrent import futures
from threading import Thread
from multiprocessing import Process

try:
    import Tkinter as tk
except ImportError:
    import tkinter as tk

def runserver():
    cmd = 'sleep 60'
    # os.system(cmd)
    subprocess.Popen(cmd, shell=True, stdout=subprocess.PIPE).stdout.read()
    print('TERMINATE BY TIMEOUT')
    return True

class WindowMain:

    def __init__(self, master):
        self.thread_srv = None
        self.master = master
        self.frame = tk.Frame(self.master)

        self.lbl = tk.Label(text="")
        self.lbl.pack()

        self.btn_quit = tk.Button(self.frame, text="Stop server", width=25, command=lambda:  self.stop_server())
        self.btn_quit.pack()

        self.button1 = tk.Button(self.frame, text="Start Server", width=25, command=lambda: self.runserver_thread())
        self.button1.pack()

        self.frame.pack()

    def stop_server(self):
        if self.thread_srv:
            print('THREAD')
            self.thread_srv.terminate()
        else:
            print("NOT THREAD")

    def runserver_thread(self):
        self.thread_srv = Process(target=runserver)
        self.thread_srv.start()

current_dir = os.path.dirname(os.path.abspath(__file__))

def quit(root):
    root.quit
    sys.exit()

def main():
    root = tk.Tk()
    tk.Button(root, text="Quitter", command=lambda: quit(root)).pack()
    app = WindowMain(root)
    root.mainloop()

if __name__ == '__main__':
    main()

解决方案(在ForceBru回答之后)

我编写了关闭django服务器的这个函数:

def close_django_runserver():
    if 'win' in sys.platform:
        pass
    else:
        cmd = 'ps aux'
        output = subprocess.Popen(cmd, shell=True, stdout=subprocess.PIPE).stdout.read()
        process_list = output.split("\n")
        for process in process_list:
            if 'manage.py' in process and 'runserver' in process:
                pid = re.sub(r'^[^\s]+\s+(\d+)\s+.*$', r'\1', process)
                subprocess.Popen('kill '+str(pid), shell=True, stdout=subprocess.PIPE).stdout.read()

它确实存在更好的解决方案,无论如何。我将在稍后编写Windows解决方案。

1 个答案:

答案 0 :(得分:2)

你运行Django启动脚本,所有内容都被初始化并设置,然后新创建的Django进程转到后台(获取* daemonized *)与线程断开连接。

所以,当你杀死那个线程时,你实际上什么也没杀,因为启动脚本完成了它的工作并且可能已经退出了。

运行启动脚本后,将创建新的Django进程。在您的机器上进行实验并找到它的名字。然后,您可以在程序中运行类似sudo killall django_process的内容。