Хорошо это - по-видимому, твердое, у меня есть pyGTK приложение, которое имеет случайные катастрофические отказы из-за ошибок X Window, которые я не могу ловить/управлять.
Таким образом, я создал обертку, которая перезапускает приложение, как только оно обнаруживает катастрофический отказ, теперь прибывает проблема, когда пользователь выходит из системы или закрывает систему, выходы приложения с состоянием 1. Но приблизительно на X ошибках это делает так также.
Таким образом, я попробовал буквально, что-либо для ловли завершения работы/выхода из системы, без успеха, вот то, что я попробовал:
import pygtk
import gtk
import sys
class Test(gtk.Window):
def delete_event(self, widget, event, data=None):
open("delete_event", "wb")
def destroy_event(self, widget, data=None):
open("destroy_event", "wb")
def destroy_event2(self, widget, event, data=None):
open("destroy_event2", "wb")
def __init__(self):
gtk.Window.__init__(self, gtk.WINDOW_TOPLEVEL)
self.show()
self.connect("delete_event", self.delete_event)
self.connect("destroy", self.destroy_event)
self.connect("destroy-event", self.destroy_event2)
def foo():
open("add_event", "wb")
def ex():
open("sys_event", "wb")
from signal import *
def clean(sig):
f = open("sig_event", "wb")
f.write(str(sig))
f.close()
exit(0)
for sig in (SIGABRT, SIGILL, SIGINT, SIGSEGV, SIGTERM):
signal(sig, lambda *args: clean(sig))
def at():
open("at_event", "wb")
import atexit
atexit.register(at)
f = Test()
sys.exitfunc = ex
gtk.quit_add(gtk.main_level(), foo)
gtk.main()
open("exit_event", "wb")
Не один из них успешно выполняется, там какой-либо низкий уровень способ обнаружить завершение работы системы? Google ничто не нашел связанным с этим.
Я предполагаю, что должен быть путь, действительно ли я прав?:/
Править: Хорошо, больше материала.
Я создал этот сценарий оболочки:
#!/bin/bash
trap test_term TERM
trap test_hup HUP
test_term(){
echo "teeeeeeeeeerm" >~/Desktop/term.info
exit 0
}
test_hup(){
echo "huuuuuuuuuuup" >~/Desktop/hup.info
exit 1
}
while [ true ]
do
echo "idle..."
sleep 2
done
И также созданный .desktop файл для выполнения его:
[Desktop Entry]
Name=Kittens
GenericName=Kittens
Comment=Kitten Script
Exec=kittens
StartupNotify=true
Terminal=false
Encoding=UTF-8
Type=Application
Categories=Network;GTK;
Name[de_DE]=Kittens
Обычно это должно создать термин файл на выходе из системы и понукать файл, когда он был запущен с и. Но не в моей Системе. GDM не заботится о сценарии вообще, когда я повторно регистрируюсь, он все еще работает.
Я также попытался использовать shopt -s huponexit
, без успеха.
EDIT2:
Также вот является еще некоторая информация aboute реальным кодом, все это похоже на это:
Wrapper Script, that catches errors and restarts the programm
-> Main Programm with GTK Mainloop
-> Background Updater Thread
Поток похож на это:
Start Wrapper
-> enter restart loop
while restarts < max:
-> start program
-> check return code
-> write error to file or exit the wrapper on 0
Теперь на завершении работы, start program
возвратитесь 1. Это означает, что или сделало hanup или завершенный родительский процесс, основная проблема состоит в том, чтобы выяснить, какой из этих двух действительно просто происходил. X Ошибок приводят к 1 также. Захват в сценарии оболочки не работает.
Если Вы хотите смотреть на фактический код, проверяют его в GitHub:
http://github.com/BonsaiDen/Atarashii
Хорошо, я наконец нашел решение :)
В этом случае нельзя полагаться на сигналы. Вы должны подключиться к Desktop Session, чтобы получать уведомления о выходе из системы.
import gnome.ui
gnome.program_init('Program', self.version) # This is going to trigger a warning that program name has been set twice, you can ignore this, it seems to be a problem with a recent version of glib, the warning is all over the place out there
client = gnome.ui.master_client() # connect us to gnome session manager, we need to init the program before this
client.connect('save-yourself', self.on_logout) # This gets called when the user confirms the logout/shutdown
client.connect('shutdown-cancelled', self.on_logout_cancel) # This gets called when the logout/shutdown is canceled
client.connect('die', self.on_logout) # Don't know when this gets called it never got in my tests
def on_logout(self, *args):
# save settings an create a file that tells the wrapper that we have exited correctly!
# we'll still return with status code 1, but that's just gtk crashing somehow
def on_logout_cancel(self, *args):
# simply delete the logout file if it exists
Одно важное замечание: не пытайтесь выйти из программы в on_logout
, если вы это сделаете, GNOME не узнает, что ваша программа была завершена, и выдаст вам диалоговое окно, в котором некоторые программы все еще работают.
Вы забыли закрыть цикл обработки событий gtk.
Этот код завершается с кодом 0
, когда вы закрываете окно:
import gtk
class Test(gtk.Window):
def destroy_event(self, widget, data=None):
gtk.main_quit()
def __init__(self):
gtk.Window.__init__(self, gtk.WINDOW_TOPLEVEL)
self.connect("destroy", self.destroy_event)
self.show()
f = Test()
gtk.main()
РЕДАКТИРОВАТЬ: Вот код для перехвата сигнала SIGTERM:
import signal
def handler(signum, frame):
print 'Signal handler called with signal', signum
print 'Finalizing main loop'
gtk.main_quit()
signal.signal(signal.SIGTERM, handler)
Остальная часть кода точно такая же, как указано выше , без изменений. Здесь он работает, когда я отправляю SIGTERM
процессу python: основной цикл gtk завершается, и программа завершается с кодом выхода 0
.