8

OK、これはおそらく難しい問題です。キャッチ/制御できない X Window エラーが原因でランダムにクラッシュする pyGTK アプリケーションがあります。

そのため、クラッシュを検出するとすぐにアプリを再起動するラッパーを作成しました。ユーザーがシステムをログアウトまたはシャットダウンすると、アプリはステータス 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 はそれに関連するものを見つけられませんでした。

方法があるに違いないと思いますよね?:/

編集: OK、もっとたくさん。

このシェルスクリプトを作成しました:

#!/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 

通常、これにより、ログアウト時にtermファイルが作成され、&で開始されたときにhupファイルが作成されます。しかし、私のシステムではありません。GDM はスクリプトをまったく気にしません。再ログインしても、まだ実行中です。

も使用してみましたがshopt -s huponexit、成功しませんでした。

EDIT2:
また、実際のコードに関する詳細情報を次に示します。全体は次のようになります。

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

ここで、シャットダウン時に 1 をstart program返します。これは、ハナップが発生したか、親プロセスが終了したかのいずれかを意味します。主な問題は、これら 2 つのうちどちらが発生したかを把握することです。X エラーも 1 になります。シェルスクリプトでのトラップは機能しません。

実際のコードを見たい場合は、GitHub でチェックしてください:
http://github.com/BonsaiDen/Atarashii

4

2 に答える 2

2

OK、私は最終的に解決策を見つけました:)

この場合、シグナルに頼ることはできません。ログアウトが行われるという通知を受け取るには、デスクトップ セッションに接続する必要があります。

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

ここで 1 つの重要な注意: でプログラムを終了しようとしないでください。終了するとon_logout、GNOME はプログラムが終了したことを認識できず、一部のプログラムがまだ実行中であるというダイアログが表示されます。

于 2010-03-25T22:18:50.567 に答える
0

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)

コードの残りの部分は上記とまったく同じで、変更はありません。Pythonプロセスに送信すると、ここで機能しSIGTERMます:gtkメインループが終了し、プログラムが終了コードで終了します0

于 2010-03-22T11:50:29.300 に答える