Please don’t consider it a duplicate before reading, There are a lot of questions about multithreading and keyboard interrupt, but i didn’t find any considering os.system and it looks like it’s important.
I have a python script which makes some external calls in worker threads.
I want it to exit if I press ctrl+c But it look like the main thread ignores it.
Something like this:
from threading import Thread
import sys
import os
def run(i):
while True:
os.system("sleep 10")
print i
def main():
threads=[]
try:
for i in range(0, 3):
threads.append(Thread(target=run, args=(i,)))
threads[i].daemon=True
threads[i].start()
for i in range(0, 3):
while True:
threads[i].join(10)
if not threads[i].isAlive():
break
except(KeyboardInterrupt, SystemExit):
sys.exit("Interrupted by ctrl+c\n")
if __name__ == '__main__':
main()
Surprisingly, it works fine if I change os.system("sleep 10") to time.sleep(10).
I’m not sure what operating system and shell you are using. I describe Mac OS X and Linux with zsh (bash/sh should act similar).
When you hit Ctrl+C, all programs running in the foreground in your current terminal receive the signal SIGINT. In your case it’s your main python process and all processes spawned by os.system.
Processes spawned by os.system then terminate their execution. Usually when python script receives SIGINT, it raises KeyboardInterrupt exception, but your main process ignores SIGINT, because of
os.system(). Pythonos.system()calls the Standard C functionsystem(), that makes calling process ignore SIGINT (man Linux / man Mac OS X).So neither of your python threads receives SIGINT, it’s only children processes who get it.
When you remove os.system() call, your python process stops ignoring SIGINT, and you get
KeyboardInterrupt.You can replace
os.system("sleep 10")withsubprocess.call(["sleep", "10"]).subprocess.call()doesn’t make your process ignore SIGINT.