Reworked mac gui to not lock up upon launch

Previously, once the node itself was launched, the UI event loop was no longer
running.  This meant that the app would sit around seemingly 'wedged' and being
reported as 'Not Responding' by the os.

This chnages that by actually implementing a wxPython gui which is left running
while the reactor, and the node within it, is launched in another thread.
Beyond 'quit' -> reactor.stop, there are no interactions between the threads.

The ui provides 'open web root' and 'open account page' actions, both in the
file menu, and in the (right click) dock icon menu.


Something weird in the handling of wxpython's per-frame menubar stuff seems to
mean that the menu bar only displays the file menu and about etc (i.e. the items
from the wx menubar) if the focus changes from and back to the app while the 
frame the menubar belongs to is displayed.  Hence a splash frame comes up at
startup to provide an opportunity.

It also seems that, in the case that the file menu is not available, that one
can induce it to reappear by choosing 'about' from the dock menu, and then 
closing the about window.
This commit is contained in:
robk-tahoe 2008-01-24 20:00:28 -07:00
parent 5085c35002
commit 2f2de9df2e
2 changed files with 205 additions and 46 deletions

View File

@ -1,6 +1,6 @@
BACKEND_URL = 'https://www-test.allmydata.com/native_client.php' BACKEND_URL = 'https://www-test.allmydata.com/native_client.php'
#REGISTER_PAGE = 'https://www-test.allmydata.com/register' ACCOUNT_PAGE = 'https://www-test.allmydata.com/account'
TAHOESVC_NAME = 'Tahoe' TAHOESVC_NAME = 'Tahoe'
WINFUSESVC_NAME = 'Allmydata Tahoe SMB' WINFUSESVC_NAME = 'Allmydata Tahoe SMB'
@ -265,7 +265,6 @@ class RegisterButtonPanel(wx.Panel):
self.SetAutoLayout(True) self.SetAutoLayout(True)
def on_reg_button(self, event): def on_reg_button(self, event):
#webbrowser.open(REGISTER_PAGE)
self.app.swap_to_register_frame() self.app.swap_to_register_frame()
class RegisterPanel(wx.Panel): class RegisterPanel(wx.Panel):

View File

@ -1,8 +1,23 @@
import sys import operator
import os import os
import stat import stat
import sys
import thread
import threading
import traceback import traceback
import urllib
import webbrowser
import wx
from twisted.internet import reactor
from twisted.python import log, logfile
import allmydata
from allmydata import client
from allmydata.gui.confwiz import ConfWizApp, ACCOUNT_PAGE
import amdicon
TRY_TO_INSTALL_TAHOE_SCRIPT = True TRY_TO_INSTALL_TAHOE_SCRIPT = True
TAHOE_SCRIPT = '''#!/bin/bash TAHOE_SCRIPT = '''#!/bin/bash
@ -15,8 +30,6 @@ fi
''' '''
def run_macapp(): def run_macapp():
import operator
basedir = os.path.expanduser('~/.tahoe') basedir = os.path.expanduser('~/.tahoe')
if not os.path.isdir(basedir): if not os.path.isdir(basedir):
app_supp = os.path.expanduser('~/Library/Application Support/Allmydata Tahoe/') app_supp = os.path.expanduser('~/Library/Application Support/Allmydata Tahoe/')
@ -24,60 +37,100 @@ def run_macapp():
os.makedirs(app_supp) os.makedirs(app_supp)
os.symlink(app_supp, basedir) os.symlink(app_supp, basedir)
if not os.path.exists(os.path.join(basedir, 'webport')): app = App(basedir)
f = file(os.path.join(basedir, 'webport'), 'wb') return app.run()
class App(object):
def __init__(self, basedir):
self.basedir = basedir
def files_exist(self, file_list):
extant_conf = [ os.path.exists(os.path.join(self.basedir, f)) for f in file_list ]
return reduce(operator.__and__, extant_conf)
def is_config_incomplete(self):
necessary_conf_files = ['introducer.furl', 'private/root_dir.cap']
need_config = not self.files_exist(necessary_conf_files)
if need_config:
print 'some config is missing from basedir (%s): %s' % (self.basedir, necessary_conf_files)
return need_config
def run(self):
# handle initial config
if not os.path.exists(os.path.join(self.basedir, 'webport')):
f = file(os.path.join(self.basedir, 'webport'), 'wb')
f.write('8123') f.write('8123')
f.close() f.close()
def files_exist(file_list): if self.is_config_incomplete():
extant_conf = [ os.path.exists(os.path.join(basedir, f)) for f in file_list ]
return reduce(operator.__and__, extant_conf)
def is_config_incomplete():
necessary_conf_files = ['introducer.furl', 'private/root_dir.cap']
need_config = not files_exist(necessary_conf_files)
if need_config:
print 'some config is missing from basedir (%s): %s' % (basedir, necessary_conf_files)
return need_config
if is_config_incomplete():
#import wx
from allmydata.gui.confwiz import ConfWizApp
app = ConfWizApp() app = ConfWizApp()
app.MainLoop() app.MainLoop()
if is_config_incomplete(): if self.is_config_incomplete():
print 'config still incomplete; confwiz cancelled, exiting' print 'config still incomplete; confwiz cancelled, exiting'
return 1 return 1
from twisted.internet import reactor
from twisted.python import log, logfile
from allmydata import client
# set up twisted logging. this will become part of the node rsn. # set up twisted logging. this will become part of the node rsn.
logdir = os.path.join(basedir, 'logs') logdir = os.path.join(self.basedir, 'logs')
if not os.path.exists(logdir): if not os.path.exists(logdir):
os.makedirs(logdir) os.makedirs(logdir)
lf = logfile.LogFile('tahoesvc.log', logdir) lf = logfile.LogFile('tahoesvc.log', logdir)
log.startLogging(lf) log.startLogging(lf)
def webopen(): if TRY_TO_INSTALL_TAHOE_SCRIPT:
if files_exist(['node.url', 'private/root_dir.cap']): self.maybe_install_tahoe_script()
# actually start up the node and the ui
os.chdir(self.basedir)
self.start_reactor()
try:
guiapp = MacGuiApp(app=self)
guiapp.MainLoop()
log.msg('gui mainloop exited')
except:
log.err()
self.stop_reactor()
return 0
def start_reactor(self):
self.reactor_shutdown = threading.Event()
thread.start_new_thread(self.launch_reactor, ())
def launch_reactor(self):
# run the node itself
c = client.Client(self.basedir)
reactor.callLater(0, c.startService) # after reactor startup
reactor.run(installSignalHandlers=False)
self.reactor_shutdown.set()
def stop_reactor(self):
# trigger reactor shutdown, and block waiting on it
reactor.callFromThread(reactor.stop)
log.msg('waiting for reactor shutdown')
self.reactor_shutdown.wait()
log.msg('reactor shut down')
def webopen(self):
if self.files_exist(['node.url', 'private/root_dir.cap']):
def read_file(f): def read_file(f):
fh = file(f, 'rb') fh = file(f, 'rb')
contents = fh.read().strip() contents = fh.read().strip()
fh.close() fh.close()
return contents return contents
import urllib, webbrowser nodeurl = read_file(os.path.join(self.basedir, 'node.url'))
nodeurl = read_file(os.path.join(basedir, 'node.url'))
if nodeurl[-1] != "/": if nodeurl[-1] != "/":
nodeurl += "/" nodeurl += "/"
root_dir = read_file(os.path.join(basedir, 'private/root_dir.cap')) root_dir = read_file(os.path.join(self.basedir, 'private/root_dir.cap'))
url = nodeurl + "uri/%s/" % urllib.quote(root_dir) url = nodeurl + "uri/%s/" % urllib.quote(root_dir)
webbrowser.open(url) webbrowser.open(url)
else: else:
print 'files missing, not opening initial webish root page' print 'files missing, not opening initial webish root page'
def maybe_install_tahoe_script(): def maybe_install_tahoe_script(self):
path_candidates = ['/usr/local/bin', '~/bin', '~/Library/bin'] path_candidates = ['/usr/local/bin', '~/bin', '~/Library/bin']
env_path = map(os.path.expanduser, os.environ['PATH'].split(':')) env_path = map(os.path.expanduser, os.environ['PATH'].split(':'))
if not sys.executable.endswith('/python'): if not sys.executable.endswith('/python'):
@ -110,18 +163,125 @@ def run_macapp():
else: else:
print 'no remaining candidate paths for installation of tahoe script' print 'no remaining candidate paths for installation of tahoe script'
if TRY_TO_INSTALL_TAHOE_SCRIPT:
maybe_install_tahoe_script()
# run the node itself def DisplayTraceback(message):
os.chdir(basedir) xc = traceback.format_exception(*sys.exc_info())
c = client.Client(basedir) wx.MessageBox(u"%s\n (%s)"%(message,''.join(xc)), 'Error')
reactor.callLater(0, c.startService) # after reactor startup
reactor.callLater(4, webopen) # give node a chance to connect before loading root dir
reactor.run()
return 0 WEBOPEN_ID = wx.NewId()
ACCOUNT_PAGE_ID = wx.NewId()
class SplashFrame(wx.Frame):
def __init__(self):
wx.Frame.__init__(self, None, -1, 'Allmydata Tahoe')
self.SetSizeHints(100, 100, 600, 800)
self.SetIcon(amdicon.getIcon())
self.Bind(wx.EVT_CLOSE, self.on_close)
background = wx.Panel(self, -1)
background.parent = self
self.login_panel = SplashPanel(background, self.on_close)
sizer = wx.BoxSizer(wx.VERTICAL)
background_sizer = wx.BoxSizer(wx.VERTICAL)
background_sizer.Add(self.login_panel, 1, wx.ALIGN_CENTER_HORIZONTAL | wx.ALL, 26)
background.SetSizer(background_sizer)
sizer.Add(background, 0, wx.EXPAND | wx.ALL, 0)
self.SetSizer(sizer)
self.SetAutoLayout(True)
self.Fit()
self.Layout()
def on_close(self, event):
self.Show(False)
class SplashPanel(wx.Panel):
def __init__(self, parent, on_close):
wx.Panel.__init__(self, parent, -1)
self.parent = parent
self.sizer = wx.BoxSizer(wx.VERTICAL)
self.label = wx.StaticText(self, -1, 'Allmydata Tahoe')
font = self.label.GetFont()
font.SetPointSize(26)
self.label.SetFont(font)
self.ver_label = wx.StaticText(self, -1, str(allmydata.__version__))
self.ok = wx.Button(self, -1, 'Ok')
self.Bind(wx.EVT_BUTTON, on_close, self.ok)
self.sizer.Add(self.label, 0, wx.CENTER | wx.ALL, 2)
self.sizer.Add(self.ver_label, 0, wx.CENTER | wx.ALL, 2)
self.sizer.Add(wx.Size(42,42), 1, wx.EXPAND | wx.ALL, 2)
self.sizer.Add(self.ok, 0, wx.CENTER | wx.ALL, 2)
self.SetSizer(self.sizer)
self.SetAutoLayout(True)
class MacGuiApp(wx.App):
def __init__(self, app):
wx.App.__init__(self)
self.app = app
def OnInit(self):
try:
self.frame = SplashFrame()
self.frame.Show(True)
self.SetTopWindow(self.frame)
wx.FutureCall(4096, self.on_timer, None)
self.setup_dock_icon()
menubar = self.setup_app_menu(self.frame)
self.frame.SetMenuBar(menubar)
return True
except:
DisplayTraceback('exception on startup')
sys.exit()
def on_timer(self, event):
self.frame.Show(False)
def setup_dock_icon(self):
self.tbicon = wx.TaskBarIcon()
self.tbicon.SetIcon(amdicon.getIcon(), "Allmydata Tahoe")
wx.EVT_TASKBAR_RIGHT_UP(self.tbicon, self.on_dock_menu)
def setup_app_menu(self, frame):
menubar = wx.MenuBar()
file_menu = wx.Menu()
item = file_menu.Append(WEBOPEN_ID, text='Open Web Root')
frame.Bind(wx.EVT_MENU, self.on_webopen, item)
item = file_menu.Append(ACCOUNT_PAGE_ID, text='Open Account Page')
frame.Bind(wx.EVT_MENU, self.on_account_page, item)
item = file_menu.Append(wx.ID_ABOUT, text='About')
frame.Bind(wx.EVT_MENU, self.on_about, item)
item = file_menu.Append(wx.ID_EXIT, text='Quit')
frame.Bind(wx.EVT_MENU, self.on_quit, item)
menubar.Append(file_menu, 'File')
return menubar
def on_dock_menu(self, event):
dock_menu = wx.Menu()
item = dock_menu.Append(wx.NewId(), text='About')
self.tbicon.Bind(wx.EVT_MENU, self.on_about, item)
item = dock_menu.Append(WEBOPEN_ID, text='Open Web Root')
self.tbicon.Bind(wx.EVT_MENU, self.on_webopen, item)
item = dock_menu.Append(ACCOUNT_PAGE_ID, text='Open Account Page')
self.tbicon.Bind(wx.EVT_MENU, self.on_account_page, item)
self.tbicon.PopupMenu(dock_menu)
def on_about(self, event):
self.frame.Show(True)
def on_quit(self, event):
self.ExitMainLoop()
def on_webopen(self, event):
self.app.webopen()
def on_account_page(self, event):
webbrowser.open(ACCOUNT_PAGE)
def main(argv): def main(argv):