mirror of
https://github.com/sshuttle/sshuttle.git
synced 2025-01-08 15:09:37 +01:00
353 lines
12 KiB
Python
353 lines
12 KiB
Python
import sys, os, pty
|
|
from AppKit import *
|
|
import my, models, askpass
|
|
|
|
def sshuttle_args(host, auto_nets, auto_hosts, nets, debug):
|
|
argv = [my.bundle_path('sshuttle/sshuttle', ''), '-r', host]
|
|
assert(argv[0])
|
|
if debug:
|
|
argv.append('-v')
|
|
if auto_nets:
|
|
argv.append('--auto-nets')
|
|
if auto_hosts:
|
|
argv.append('--auto-hosts')
|
|
argv += nets
|
|
return argv
|
|
|
|
|
|
class _Callback(NSObject):
|
|
def initWithFunc_(self, func):
|
|
self = super(_Callback, self).init()
|
|
self.func = func
|
|
return self
|
|
def func_(self, obj):
|
|
return self.func(obj)
|
|
|
|
|
|
class Callback:
|
|
def __init__(self, func):
|
|
self.obj = _Callback.alloc().initWithFunc_(func)
|
|
self.sel = self.obj.func_
|
|
|
|
|
|
class Runner:
|
|
def __init__(self, argv, logfunc, promptfunc, serverobj):
|
|
print 'in __init__'
|
|
self.id = argv
|
|
self.rv = None
|
|
self.pid = None
|
|
self.fd = None
|
|
self.logfunc = logfunc
|
|
self.promptfunc = promptfunc
|
|
self.serverobj = serverobj
|
|
self.buf = ''
|
|
self.logfunc('\nConnecting to %s.\n' % self.serverobj.host())
|
|
print 'will run: %r' % argv
|
|
self.serverobj.setConnected_(False)
|
|
pid,fd = pty.fork()
|
|
if pid == 0:
|
|
# child
|
|
try:
|
|
os.execvp(argv[0], argv)
|
|
except Exception, e:
|
|
sys.stderr.write('failed to start: %r\n' % e)
|
|
raise
|
|
finally:
|
|
os._exit(42)
|
|
# parent
|
|
self.pid = pid
|
|
self.file = NSFileHandle.alloc()\
|
|
.initWithFileDescriptor_closeOnDealloc_(fd, True)
|
|
self.cb = Callback(self.gotdata)
|
|
NSNotificationCenter.defaultCenter()\
|
|
.addObserver_selector_name_object_(self.cb.obj, self.cb.sel,
|
|
NSFileHandleDataAvailableNotification, self.file)
|
|
self.file.waitForDataInBackgroundAndNotify()
|
|
|
|
def __del__(self):
|
|
self.wait()
|
|
|
|
def _try_wait(self, options):
|
|
if self.rv == None and self.pid > 0:
|
|
pid,code = os.waitpid(self.pid, options)
|
|
if pid == self.pid:
|
|
if os.WIFEXITED(code):
|
|
self.rv = os.WEXITSTATUS(code)
|
|
else:
|
|
self.rv = -os.WSTOPSIG(code)
|
|
self.serverobj.setConnected_(False)
|
|
self.serverobj.setError_('VPN process died')
|
|
self.logfunc('Disconnected.\n')
|
|
print 'wait_result: %r' % self.rv
|
|
return self.rv
|
|
|
|
def wait(self):
|
|
return self._try_wait(0)
|
|
|
|
def poll(self):
|
|
return self._try_wait(os.WNOHANG)
|
|
|
|
def kill(self):
|
|
assert(self.pid > 0)
|
|
print 'killing: pid=%r rv=%r' % (self.pid, self.rv)
|
|
if self.rv == None:
|
|
self.logfunc('Disconnecting from %s.\n' % self.serverobj.host())
|
|
os.kill(self.pid, 15)
|
|
self.wait()
|
|
|
|
def gotdata(self, notification):
|
|
print 'gotdata!'
|
|
d = str(self.file.availableData())
|
|
if d:
|
|
self.logfunc(d)
|
|
self.buf = self.buf + d
|
|
if 'Connected.\r\n' in self.buf:
|
|
self.serverobj.setConnected_(True)
|
|
self.buf = self.buf[-4096:]
|
|
if self.buf.strip().endswith(':'):
|
|
lastline = self.buf.rstrip().split('\n')[-1]
|
|
resp = self.promptfunc(lastline)
|
|
add = ' (response)\n'
|
|
self.buf += add
|
|
self.logfunc(add)
|
|
self.file.writeData_(my.Data(resp + '\n'))
|
|
self.file.waitForDataInBackgroundAndNotify()
|
|
self.poll()
|
|
#print 'gotdata done!'
|
|
|
|
|
|
class SshuttleApp(NSObject):
|
|
def initialize(self):
|
|
d = my.PList('UserDefaults')
|
|
my.Defaults().registerDefaults_(d)
|
|
|
|
|
|
class SshuttleController(NSObject):
|
|
# Interface builder outlets
|
|
startAtLoginField = objc.IBOutlet()
|
|
autoReconnectField = objc.IBOutlet()
|
|
debugField = objc.IBOutlet()
|
|
routingField = objc.IBOutlet()
|
|
prefsWindow = objc.IBOutlet()
|
|
serversController = objc.IBOutlet()
|
|
logField = objc.IBOutlet()
|
|
|
|
servers = []
|
|
conns = {}
|
|
|
|
def _connect(self, server):
|
|
host = server.host()
|
|
print 'connecting %r' % host
|
|
self.fill_menu()
|
|
def logfunc(msg):
|
|
print 'log! (%d bytes)' % len(msg)
|
|
self.logField.textStorage()\
|
|
.appendAttributedString_(NSAttributedString.alloc()\
|
|
.initWithString_(msg))
|
|
self.logField.didChangeText()
|
|
def promptfunc(prompt):
|
|
print 'prompt! %r' % prompt
|
|
return askpass.askpass(prompt)
|
|
nets_mode = server.autoNets()
|
|
if nets_mode == models.NET_MANUAL:
|
|
manual_nets = ["%s/%d" % (i.subnet(), i.width())
|
|
for i in server.nets()]
|
|
elif nets_mode == models.NET_ALL:
|
|
manual_nets = ['0/0']
|
|
else:
|
|
manual_nets = []
|
|
conn = Runner(sshuttle_args(host,
|
|
auto_nets = nets_mode == models.NET_AUTO,
|
|
auto_hosts = server.autoHosts(),
|
|
nets = manual_nets,
|
|
debug = self.debugField.state()),
|
|
logfunc=logfunc, promptfunc=promptfunc,
|
|
serverobj=server)
|
|
self.conns[host] = conn
|
|
|
|
def _disconnect(self, server):
|
|
host = server.host()
|
|
print 'disconnecting %r' % host
|
|
conn = self.conns.get(host)
|
|
if conn:
|
|
conn.kill()
|
|
self.fill_menu()
|
|
self.logField.textStorage().setAttributedString_(
|
|
NSAttributedString.alloc().initWithString_(''))
|
|
|
|
@objc.IBAction
|
|
def cmd_connect(self, sender):
|
|
server = sender.representedObject()
|
|
server.setWantConnect_(True)
|
|
|
|
@objc.IBAction
|
|
def cmd_disconnect(self, sender):
|
|
server = sender.representedObject()
|
|
server.setWantConnect_(False)
|
|
|
|
@objc.IBAction
|
|
def cmd_show(self, sender):
|
|
self.prefsWindow.makeKeyAndOrderFront_(self)
|
|
NSApp.activateIgnoringOtherApps_(True)
|
|
|
|
@objc.IBAction
|
|
def cmd_quit(self, sender):
|
|
NSApp.performSelector_withObject_afterDelay_(NSApp.terminate_,
|
|
None, 0.0)
|
|
|
|
def fill_menu(self):
|
|
menu = self.menu
|
|
menu.removeAllItems()
|
|
|
|
def additem(name, func, obj):
|
|
it = menu.addItemWithTitle_action_keyEquivalent_(name, None, "")
|
|
it.setRepresentedObject_(obj)
|
|
it.setTarget_(self)
|
|
it.setAction_(func)
|
|
def addnote(name):
|
|
additem(name, None, None)
|
|
|
|
any_inprogress = None
|
|
any_conn = None
|
|
any_err = None
|
|
if len(self.servers):
|
|
for i in self.servers:
|
|
host = i.host()
|
|
want = i.wantConnect()
|
|
connected = i.connected()
|
|
numnets = len(list(i.nets()))
|
|
if not host:
|
|
additem('Connect Untitled', None, i)
|
|
elif i.autoNets() == models.NET_MANUAL and not numnets:
|
|
additem('Connect %s (no routes)' % host, None, i)
|
|
elif want:
|
|
any_conn = i
|
|
additem('Disconnect %s' % host, self.cmd_disconnect, i)
|
|
else:
|
|
additem('Connect %s' % host, self.cmd_connect, i)
|
|
if not want:
|
|
msg = 'Off'
|
|
elif i.error():
|
|
msg = 'ERROR - try reconnecting'
|
|
any_err = i
|
|
elif connected:
|
|
msg = 'Connected'
|
|
else:
|
|
msg = 'Connecting...'
|
|
any_inprogress = i
|
|
addnote(' State: %s' % msg)
|
|
if i.autoNets() == 0:
|
|
addnote(' Routes: All')
|
|
elif i.autoNets() == 2:
|
|
addnote(' Routes: Auto')
|
|
else:
|
|
addnote(' Routes: Custom')
|
|
else:
|
|
addnote('No servers defined yet')
|
|
|
|
menu.addItem_(NSMenuItem.separatorItem())
|
|
additem('Preferences...', self.cmd_show, None)
|
|
additem('Quit Sshuttle VPN', self.cmd_quit, None)
|
|
|
|
if any_err:
|
|
self.statusitem.setImage_(self.img_err)
|
|
self.statusitem.setTitle_('Error!')
|
|
elif any_conn:
|
|
self.statusitem.setImage_(self.img_running)
|
|
if any_inprogress:
|
|
self.statusitem.setTitle_('Connecting...')
|
|
else:
|
|
self.statusitem.setTitle_('')
|
|
else:
|
|
self.statusitem.setImage_(self.img_idle)
|
|
self.statusitem.setTitle_('')
|
|
|
|
def load_servers(self):
|
|
l = my.Defaults().arrayForKey_('servers') or []
|
|
sl = []
|
|
for s in l:
|
|
host = s.get('host', None)
|
|
if not host: continue
|
|
|
|
nets = s.get('nets', [])
|
|
nl = []
|
|
for n in nets:
|
|
subnet = n[0]
|
|
width = n[1]
|
|
net = models.SshuttleNet.alloc().init()
|
|
net.setSubnet_(subnet)
|
|
net.setWidth_(width)
|
|
nl.append(net)
|
|
|
|
autoNets = s.get('autoNets', 1)
|
|
autoHosts = s.get('autoHosts', 1)
|
|
srv = models.SshuttleServer.alloc().init()
|
|
srv.setHost_(host)
|
|
srv.setAutoNets_(autoNets)
|
|
srv.setAutoHosts_(autoHosts)
|
|
srv.setNets_(nl)
|
|
sl.append(srv)
|
|
self.serversController.addObjects_(sl)
|
|
self.serversController.setSelectionIndex_(0)
|
|
|
|
def save_servers(self):
|
|
l = []
|
|
for s in self.servers:
|
|
host = s.host()
|
|
if not host: continue
|
|
nets = []
|
|
for n in s.nets():
|
|
subnet = n.subnet()
|
|
if not subnet: continue
|
|
nets.append((subnet, n.width()))
|
|
d = dict(host=s.host(),
|
|
nets=nets,
|
|
autoNets=s.autoNets(),
|
|
autoHosts=s.autoHosts())
|
|
l.append(d)
|
|
my.Defaults().setObject_forKey_(l, 'servers')
|
|
self.fill_menu()
|
|
|
|
def awakeFromNib(self):
|
|
self.routingField.removeAllItems()
|
|
tf = self.routingField.addItemWithTitle_
|
|
tf('Send all traffic through this server')
|
|
tf('Determine automatically')
|
|
tf('Custom...')
|
|
|
|
# Hmm, even when I mark this as !enabled in the .nib, it still comes
|
|
# through as enabled. So let's just disable it here (since we don't
|
|
# support this feature yet).
|
|
self.startAtLoginField.setEnabled_(False)
|
|
self.startAtLoginField.setState_(False)
|
|
self.autoReconnectField.setEnabled_(False)
|
|
self.autoReconnectField.setState_(False)
|
|
|
|
self.load_servers()
|
|
|
|
# Initialize our menu item
|
|
self.menu = NSMenu.alloc().initWithTitle_('Sshuttle')
|
|
bar = NSStatusBar.systemStatusBar()
|
|
statusitem = bar.statusItemWithLength_(NSVariableStatusItemLength)
|
|
self.statusitem = statusitem
|
|
self.img_idle = my.Image('chicken-tiny-bw', 'png')
|
|
self.img_running = my.Image('chicken-tiny', 'png')
|
|
self.img_err = my.Image('chicken-tiny-err', 'png')
|
|
statusitem.setImage_(self.img_idle)
|
|
statusitem.setHighlightMode_(True)
|
|
statusitem.setMenu_(self.menu)
|
|
self.fill_menu()
|
|
|
|
models.configchange_callback = my.DelayedCallback(self.save_servers)
|
|
|
|
def sc(server):
|
|
if server.wantConnect():
|
|
self._connect(server)
|
|
else:
|
|
self._disconnect(server)
|
|
models.setconnect_callback = sc
|
|
|
|
|
|
# Note: NSApplicationMain calls sys.exit(), so this never returns.
|
|
NSApplicationMain(sys.argv)
|