Portál AbcLinuxu, 11. května 2025 10:09

Dotaz: python gui mainloop a dalsi funkce

12.6.2006 21:14 Jáchym Čepický | skóre: 29 | blog: U_Jachyma
python gui mainloop a dalsi funkce
Přečteno: 351×
Odpovědět | Admin
Zdravím,

potřeboval bych, aby se změna v nějakém souboru projevila v GUI aplikaci (v podstatě tail -f). Zkoušel jsem to přes fork a semafor, ale dostal jsem tip, že by se to mělo nějak dát udělat přes MainLoop() - to je koneckonců smyčka, která by měla zavolat určitou funkci kontrolující soubor.

Ale zatím jsem nenašel "obecný" postup, jak toho dosáhnout. Programátor nejsem, netuším, po čem vlastně pátrám.

Používám wxPython (ale postup asi bude obecný pro jaký koliv toolkit), část kódu:
class Monitor(wx.App):
    def OnInit(self):
        init GUI
    [....]
     
    def watcher(self):
        if zmena_v_souboru:
           zmena_v_gui

def main():
    pymonitor = Monitor(0)
    pymonitor.MainLoop()

main()
Jak na to?

Díky
Nástroje: Začni sledovat (1) ?Zašle upozornění na váš email při vložení nového komentáře.

Odpovědi

12.6.2006 22:45 Michal Čihař | skóre: 61 | blog: Bláboly | Praha
Rozbalit Rozbalit vše Re: python gui mainloop a dalsi funkce
Odpovědět | | Sbalit | Link | Blokovat | Admin
tail -f ve wxPythonu jsem potřeboval ve Wammu, třeba se tam můžeš inspirovat :-).
Weblate - překládání přes web | Gammu SMSD - posílání SMS | Blog
12.6.2006 22:49 Petr Mach
Rozbalit Rozbalit vše Re: python gui mainloop a dalsi funkce
Odpovědět | | Sbalit | Link | Blokovat | Admin
Nepochopil jsem, na co se vlastne ptas. Poloz konkretni dotazy. Napr. 1) jak poznat zmenu v souboru?

Odp: podle jeho casu

2) ... (dal se ptej ty) ...
12.6.2006 23:36 Petr Mach
Rozbalit Rozbalit vše Re: python gui mainloop a dalsi funkce
No, mozna uz chapu. Kdybys pouzival pygtk, asi bys hledal tohle:

http://www.pygtk.org/pygtk2reference/gobject-functions.html#function-gobject--timeout-add
13.6.2006 08:39 Jan Martinek | skóre: 43 | blog: johny | Brno
Rozbalit Rozbalit vše Re: python gui mainloop a dalsi funkce
Jééé! Zrovna jsem nedávno chtěl udělat jakési odpočítávání a uvízl jsem. Přesně tohle potřebuju :-)
Tušil jsem, že něco takového existuje, ale hledal jsem to pod gtk místo gobject.
13.6.2006 09:13 Jan Martinek | skóre: 43 | blog: johny | Brno
Rozbalit Rozbalit vše Re: python gui mainloop a dalsi funkce
Škoda, že se nedá nějak udělat, aby se automaticky vyvolala událost při změně souboru. To periodické kontrolování každou sekundu se mi moc nelíbí. Ale 'tail -f' to tak dělá - nejspíš proto, aby fungoval i na topinkovači. Problém je, že select() nelze pro normální soubory použít, protože ten vždy vrátí, že jsou data k dispozici (přesněji řečeno, že read() nebude blokovat). S tím pravděpodobně souvisí i to, že pythonní gobject.io_add_watch taktéž nelze použít.

Ale existují i jiná jaderná volání - inotify nebo dnotify nebo jak se to jmenuje. Používá to fam (File Alternation Monitor) a jedna z verzí locate.

Použít se dají také jaderné leases (viz man fcntl). Zkusil jsem pomocí toho napsat 'tail -f'. Jakž takž to funguje, ale leases jsou asi moc rychlé - dostanu signál ještě než se data stihnou zapsat.
#!/usr/bin/env python
import sys, os, select, fcntl, signal

SEEK_SET, SEEK_CUR, SEEK_END = 0, 1, 2
bufsize = 4096

def handler_SIGIO(a, b):
    pass

def dump_to_end(fd):
    while True:
        chunk = os.read(fd, bufsize)
        if not chunk: break
        sys.stdout.write(chunk)
        

fd = os.open(sys.argv[1], os.O_RDONLY)

signal.signal(signal.SIGIO, handler_SIGIO)

filepos = 0
while True:
    os.lseek(fd, filepos, SEEK_SET)
    dump_to_end(fd)
    filepos = os.lseek(fd, 0, SEEK_CUR)
    print filepos   
    fcntl.fcntl(fd, fcntl.F_SETLEASE, fcntl.F_RDLCK)
    signal.pause()
Dá se to udělat i jinak? Nevýhodou je to šaškování se signálem a volání pause() :-(
13.6.2006 09:57 Jáchym Čepický | skóre: 29 | blog: U_Jachyma
Rozbalit Rozbalit vše Re: python gui mainloop a dalsi funkce
No, můj problém je, že to musí fungovat skutečně na všem (win/lin/mac) a proto se na žádná jadrná volání spoléhat nemůžu.

Ještě jsem tak přemýšlel, jít na to přes os.fork(), ale vypadá to, že vzniklé procesy nemají sdílený paměťový prostor - takže si nemůžu předávat události přes semafory (nemá někdo link na příklad s pythoním fork, ve kterém by prokazatelně dva procesy měnily jednu proměnnou?)

Díky
13.6.2006 10:27 Michal Vyskočil | skóre: 60 | blog: miblog | Praha
Rozbalit Rozbalit vše Re: python gui mainloop a dalsi funkce
Procesy už z definice nemají sdílené paměťové prostory (snad na Windows 9x) ;-), pokud si je explicitně neuděláš (třeba přes rouru). Není lepší použít vlákna?
When your hammer is C++, everything begins to look like a thumb.
13.6.2006 10:52 Jan Martinek | skóre: 43 | blog: johny | Brno
Rozbalit Rozbalit vše Re: python gui mainloop a dalsi funkce
Když se smířím s periodickým voláním stat(), tak to jde udělat docela jednoduše - bez nějakých forků či rour:
#!/usr/bin/env python
import pygtk
pygtk.require('2.0')
import gtk, gobject, stat, os, sys

class WatchFile:    
    def __init__(self, filename):

        self.filename = filename
        self.window = gtk.Window(gtk.WINDOW_TOPLEVEL)
        self.window.connect('delete_event', self.delete_event)
        self.window.connect('destroy', self.quit)
        
        self.checkbut = gtk.CheckButton('File "%s" has changed' %filename)
        self.checkbut.show()
        
        self.times = (None, None, None)
        self.has_changed()

        gobject.timeout_add(1000, self.periodic_check, None)

        self.window.add(self.checkbut)
        self.window.show()

    def periodic_check(self, object):
        if self.has_changed():
            self.checkbut.set_active(1)
        return True

    def has_changed(self):
        sr = os.lstat(self.filename)
        t = (sr.st_atime, sr.st_mtime, sr.st_ctime)
        if t != self.times:
            self.times = t
            return True
        else:
            return False

    def delete_event(self, widget, data):
        return False
    
    def quit(self, widget, data = None):
        gtk.main_quit()        

    def main(self):
        gtk.main()

if __name__ == '__main__':
    wtch = WatchFile(sys.argv[1])
    wtch.main()
Tohle ukáže checkbox, který se automaticky zaškrtne při každé změně souboru. Odšktrnutí jsem ponechal na uživateli, ať si klikne myší. Každou sekundu to kontroluje všechny tři časy u souboru. Jádrem pudla je timeout_add, o kterém psal Petr Mach.
13.6.2006 11:19 Jáchym Čepický | skóre: 29 | blog: U_Jachyma
Rozbalit Rozbalit vše Re: python gui mainloop a dalsi funkce
Jj, to je v podstatě ono, akorat pro wxpython to vypadá asi takhle:
        self.timer = wx.wxPyTimer(self.watcher)
        self.timer.Start(1000)
        self.watcher() # to je ta funkce, která  to celé dělá

čerpal jsem z roota (cirkus s okny)

dík za nasměrování správným směrem.

Založit nové vláknoNahoru

Tiskni Sdílej: Linkuj Jaggni to Vybrali.sme.sk Google Del.icio.us Facebook

ISSN 1214-1267, (c) 1999-2007 Stickfish s.r.o.