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

View File

@ -1,8 +1,23 @@
import sys
import operator
import os
import stat
import sys
import thread
import threading
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
TAHOE_SCRIPT = '''#!/bin/bash
@ -15,8 +30,6 @@ fi
'''
def run_macapp():
import operator
basedir = os.path.expanduser('~/.tahoe')
if not os.path.isdir(basedir):
app_supp = os.path.expanduser('~/Library/Application Support/Allmydata Tahoe/')
@ -24,60 +37,100 @@ def run_macapp():
os.makedirs(app_supp)
os.symlink(app_supp, basedir)
if not os.path.exists(os.path.join(basedir, 'webport')):
f = file(os.path.join(basedir, 'webport'), 'wb')
f.write('8123')
f.close()
app = App(basedir)
return app.run()
def files_exist(file_list):
extant_conf = [ os.path.exists(os.path.join(basedir, f)) for f in file_list ]
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():
def is_config_incomplete(self):
necessary_conf_files = ['introducer.furl', 'private/root_dir.cap']
need_config = not files_exist(necessary_conf_files)
need_config = not self.files_exist(necessary_conf_files)
if need_config:
print 'some config is missing from basedir (%s): %s' % (basedir, necessary_conf_files)
print 'some config is missing from basedir (%s): %s' % (self.basedir, necessary_conf_files)
return need_config
if is_config_incomplete():
#import wx
from allmydata.gui.confwiz import ConfWizApp
app = ConfWizApp()
app.MainLoop()
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.close()
if is_config_incomplete():
print 'config still incomplete; confwiz cancelled, exiting'
return 1
if self.is_config_incomplete():
app = ConfWizApp()
app.MainLoop()
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.
logdir = os.path.join(basedir, 'logs')
if not os.path.exists(logdir):
os.makedirs(logdir)
lf = logfile.LogFile('tahoesvc.log', logdir)
log.startLogging(lf)
if self.is_config_incomplete():
print 'config still incomplete; confwiz cancelled, exiting'
return 1
def webopen():
if files_exist(['node.url', 'private/root_dir.cap']):
# set up twisted logging. this will become part of the node rsn.
logdir = os.path.join(self.basedir, 'logs')
if not os.path.exists(logdir):
os.makedirs(logdir)
lf = logfile.LogFile('tahoesvc.log', logdir)
log.startLogging(lf)
if TRY_TO_INSTALL_TAHOE_SCRIPT:
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):
fh = file(f, 'rb')
contents = fh.read().strip()
fh.close()
return contents
import urllib, webbrowser
nodeurl = read_file(os.path.join(basedir, 'node.url'))
nodeurl = read_file(os.path.join(self.basedir, 'node.url'))
if nodeurl[-1] != "/":
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)
webbrowser.open(url)
else:
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']
env_path = map(os.path.expanduser, os.environ['PATH'].split(':'))
if not sys.executable.endswith('/python'):
@ -110,18 +163,125 @@ def run_macapp():
else:
print 'no remaining candidate paths for installation of tahoe script'
if TRY_TO_INSTALL_TAHOE_SCRIPT:
maybe_install_tahoe_script()
# run the node itself
os.chdir(basedir)
c = client.Client(basedir)
reactor.callLater(0, c.startService) # after reactor startup
reactor.callLater(4, webopen) # give node a chance to connect before loading root dir
reactor.run()
def DisplayTraceback(message):
xc = traceback.format_exception(*sys.exc_info())
wx.MessageBox(u"%s\n (%s)"%(message,''.join(xc)), 'Error')
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):