| 			 
 
#! /usr/bin/env python
 
 
import os, sys
 
DATADIR = os.path.realpath(sys.argv[0])
 
for i in range(2):
 
	DATADIR = os.path.split(DATADIR)[0]
 
DEFAULT_TEMPLATE = os.path.join(DATADIR, 'share', 'museek', 'museekd', 'config.xml.tmpl')
 
if not os.path.exists(DEFAULT_TEMPLATE):
 
	t_path = os.path.join('/usr', 'share', 'museek', 'museekd', 'config.xml.tmpl')
 
	if os.path.exists(t_path):
 
		DEFAULT_TEMPLATE = t_path
 
DEFAULT_CONFIG = os.path.join('~', '.museekd', 'config.xml')
 
 
from xml.dom import minidom
 
import time, stat, string, pwd
 
 
config = {}
 
 
def readDomain(node):
 
	domain = {}
 
	child = node.firstChild
 
	while child:
 
		if child.nodeName == u'key':
 
			id = child.getAttribute('id')
 
			if child.firstChild:
 
				domain[id] = child.firstChild.nodeValue
 
			else:
 
				domain[id] = ''
 
		child = child.nextSibling
 
	return domain
 
	
 
def readConfig(path = 'config.xml'):
 
	global config
 
	doc = minidom.parse(path)
 
	root = doc.firstChild
 
	node = root.firstChild
 
	while node:
 
		if node.nodeName == u'domain':
 
			id = node.getAttribute('id')
 
			config[id] = readDomain(node)
 
		node = node.nextSibling
 
 
def readTemplate(path = 'config.xml.tmpl', config_path = 'config.xml'):
 
	global config
 
	readConfig(path)
 
	if config_path.rfind('.') > config_path.rfind('/'):
 
		config_path = '.'.join(config_path.split('.')[:-1])
 
	
 
	user = pwd.getpwuid(os.getuid())[0]
 
	for domain in config.values():
 
		for key in domain.keys():
 
			value = domain[key].replace('$(USER)', user).replace('$(CONFIG)', config_path)
 
			del domain[key]
 
			key = key.replace('$(USER)', user).replace('$(CONFIG)', config_path)
 
			domain[key] = value
 
	
 
def writeConfig(path = 'newconfig.xml'):
 
	global config
 
	doc = minidom.Document()
 
	root = doc.appendChild(doc.createElement('museekd'))
 
	for i in config.keys():
 
		root.appendChild(doc.createTextNode('\n  '))
 
		domain = root.appendChild(doc.createElement('domain'))
 
		domain.setAttribute('id', i)
 
		for j in config[i].keys():
 
			domain.appendChild(doc.createTextNode('\n    '))
 
			key = domain.appendChild(doc.createElement('key'))
 
			key.setAttribute('id', j)
 
			v = config[i][j]
 
			if v:
 
				key.appendChild(doc.createTextNode(v))
 
		domain.appendChild(doc.createTextNode('\n  '))
 
	root.appendChild(doc.createTextNode('\n'))
 
	
 
	try:
 
		os.makedirs(os.path.split(path)[0])
 
	except OSError:
 
		pass
 
	
 
	f = open(path, 'w')
 
	doc.writexml(f)
 
	f.close()
 
	os.chmod(path, stat.S_IRUSR | stat.S_IWUSR)
 
	print
 
	print 'Configuration updated. Don\'t forget to restart museekd.'
 
	print
 
	
 
def choice(max):
 
	print
 
	print 'Pick a number, any number: ',
 
	line = sys.stdin.readline().strip()
 
	print
 
	try:
 
		choice = int(line)
 
	except:
 
		print 'I said, a number.. when did \'%s\' become a number?' % line
 
		time.sleep(3)
 
		return -1
 
	if choice not in range(max+1):
 
		print 'I know I said any number, but a number which actually\nhas an action assigned to it might be a wiser choice'
 
		time.sleep(3)
 
		return -1
 
	return choice
 
 
def port_choice(default):
 
	print 'Port [%s]: ' % (default),
 
	s = sys.stdin.readline().strip()
 
	print
 
	if s:
 
		try:
 
			port = int(s)
 
			return s
 
		except:
 
			print 'Ports are usually numbers'
 
			time.sleep(3)
 
	return default
 
 
def string_choice(title, default):
 
	print '%s [%s]: ' % (title, default),
 
	s = sys.stdin.readline().strip()
 
	if s:
 
		return s
 
	return default
 
 
def path_choice(default):
 
	print 'Path: ',
 
	s = sys.stdin.readline().strip()
 
	return os.path.expanduser(s)
 
 
def server():
 
	global config
 
	if not config.has_key('server'):
 
		config['server'] = {}
 
	_server = config['server'].get('host', 'server.slsknet.org')
 
	_port = config['server'].get('port', '2240')
 
	_username = config['server'].get('username', '')
 
	_password = config['server'].get('password', '')
 
	print
 
	print
 
	print 'Soulseek server setup'
 
	print
 
	print '1. Server:', _server
 
	print '2. Port:  ', _port
 
	print
 
	print '3. Username:', _username
 
	print '4. Password:', '*' * len(_password)
 
	print
 
	print '0. Leave this menu'
 
	c = choice(4)
 
	if c == 0:
 
		return 1
 
	elif c == 1:
 
		config['server']['host'] = string_choice('Server', _server)
 
	elif c == 2:
 
		config['server']['port'] = port_choice(_port)
 
	elif c == 3:
 
		config['server']['username'] = string_choice('Username', _username)
 
	elif c == 4:
 
		config['server']['password'] = string_choice('Password', _password)
 
	return 0
 
 
def connectmode():
 
	print
 
	print
 
	print 'Peer connect mode setup'
 
	print
 
	print '1. Active'
 
	print '   (use this if your client port can\'t be reached'
 
	print '    from the outside world, think firewalls / routers)'
 
	print
 
	print '2. Passive'
 
	print '   (use this if you can receive direct connections on'
 
	print '    your client port from the outside world)'
 
	print
 
	print '0. Don\'t change this setting'
 
	c = choice(2)
 
	if c == -1:
 
		return 0
 
	elif c == 1:
 
		config['clients']['connectmode'] = 'active'
 
	elif c == 2:
 
		config['clients']['connectmode'] = 'passive'
 
	return 1
 
 
def buddyshares():
 
	print
 
	print 'Special Shares List for Buddies'
 
	print
 
	print "Note: If you want to share all the files in your \"Normal\" shares"
 
	print "within the special shares, you will have to add them."	
 
	print
 
	print '1. True'
 
	print '   (use this if your want to have a seperate list for'
 
	print '    buddies that includes file normal users will not get)'
 
	print
 
	print '2. False'
 
	print '   (use this if you want everyone to get have access'
 
	print '    to a single \"Normal\" shares list)'
 
	print
 
	print '0. Don\'t change this setting'
 
	c = choice(2)
 
	if c == -1:
 
		return 0
 
	elif c == 1:
 
		config['transfers']['have_buddy_shares'] = 'true'
 
	elif c == 2:
 
		config['transfers']['have_buddy_shares'] = 'false'
 
	return 1
 
	
 
def portrange():
 
	global config
 
	if not config.has_key('clients.bind'):
 
		config['clients.bind'] = {}
 
	_first = config['clients.bind'].get('first', '2234')
 
	_last = config['clients.bind'].get('last', '2240')
 
	if not config.has_key('clients'):
 
		config['clients'] = {}
 
	_mode = config['clients'].get('connectmode', 'active')
 
	print
 
	print
 
	print 'Client port range setup'
 
	print
 
	print '1. First port to try: ', _first
 
	print '2. Last port to try:  ', _last
 
	print
 
	print '3. Connection mode:   ', _mode
 
	print
 
	print '0. Leave this menu'
 
	c = choice(3)
 
	if c == 1:
 
		config['clients.bind']['first'] = port_choice(_first)
 
	elif c == 2:
 
		config['clients.bind']['last'] = port_choice(_last)
 
	elif c == 3:
 
		while not connectmode(): pass
 
	else:
 
		return 1
 
	return 0
 
	
 
def dirs():
 
	if not config.has_key('transfers'):
 
		config['transfers'] = {}
 
	_complete = config['transfers'].get('download-dir', '~/downloads')
 
	_incomplete = config['transfers'].get('incomplete-dir', '')
 
 
	print
 
	print
 
	print 'Incomplete and completed download setup'
 
	print 'Please input the full directory path, or use ~/ for your home directory'
 
	print 
 
	print '1. Completed downloads:', _complete
 
	print '2. Incomplete downloads:', _incomplete
 
	print '   (if this is empty, completed download dirs will be used)'
 
	print
 
	print '0. Leave this menu'
 
	c = choice(2)
 
	if c == 1:
 
		config['transfers']['download-dir'] = path_choice(_complete)
 
	elif c == 2:
 
		config['transfers']['incomplete-dir'] = path_choice(_incomplete)
 
	else:
 
		return 1
 
	return 0
 
 
def charsets():
 
	if not config.has_key('encoding'):
 
		config['encoding'] = {}
 
	_network = config['encoding'].get('network', 'UTF-8')
 
	_filesystem = config['encoding'].get('filesystem', 'LATIN1')
 
	print
 
	print
 
	print 'Network and filesystem character set setup'
 
	print
 
	print '1. Network character set:   ', _network
 
	print '   (Make the world a better place, use UTF-8!)'
 
	print
 
	print '2. Filesystem character set:', _filesystem
 
	print '   (latin1 is usually a good choice)'
 
	print
 
	print '0. Leave this menu'
 
	c = choice(2)
 
	if c == 1:
 
		config['encoding']['network'] = string_choice('Character set', _network)
 
	elif c == 2:
 
		config['encoding']['filesystem'] = string_choice('Character set', _filesystem)
 
	else:
 
		return 1
 
	return 0
 
	
 
def new_iface():
 
	print
 
	print
 
	print 'New interface listener setup'
 
	print
 
	print '1. Add TCP socket listener'
 
	print '2. Add UNIX socket listener'
 
	print
 
	print '0. Don\'t add a new interface listener'
 
	c = choice(2)
 
	if c == -1:
 
		return 0
 
	elif c == 1:
 
		host = string_choice('Local address to bind to (empty = all)', '')
 
		port = port_choice('')
 
		if not port:
 
			print 'A port is a requirement to be able to bind...'
 
			time.sleep(3)
 
			return 0
 
		config['interfaces.bind'][host + ':' + port] = ''
 
	elif c == 2:
 
		path = string_choice('Path + filename of UNIX socket', '')
 
		if not path:
 
			return 0
 
		if path[0] != '/':
 
			print 'UNIX socket pathnames have to be absolute'
 
			time.sleep(3)
 
			return 0
 
		config['interfaces.bind'][path] = ''
 
	return 1
 
	
 
def ifaces():
 
	if not config.has_key('interfaces'):
 
		config['interfaces'] = {}
 
	_password = config['interfaces'].get('password', '')
 
	if not config.has_key('interfaces.bind'):
 
		config['interfaces.bind'] = {}
 
	_ifaces = config['interfaces.bind'].keys()
 
	_ifaces.sort()
 
	print
 
	print
 
	print 'Interface listener setup'
 
	print
 
	print '1. Interface password', '*' * len(_password)
 
	print
 
	print '2. Add new interface listener'
 
	print
 
	ix = 2
 
	for i in _ifaces:
 
		ix = ix + 1
 
		print '%i. Delete interface listener' % (ix), i 
 
	if not _ifaces:
 
		print '(no interface listeners defined)'
 
	print
 
	print '0. Leave this menu'
 
	c = choice(ix)
 
	if c == 0:
 
		return 1
 
	elif c == 1:
 
		config['interfaces']['password'] = string_choice('Password', _password)
 
	elif c == 2:
 
		while not new_iface(): pass
 
	elif c != -1:
 
		del config['interfaces.bind'][_ifaces[c - 3]]
 
	return 0
 
 
def escapeCommand(filename):
 
	"""Escapes special characters for command execution"""
 
	escaped = ""
 
	for ch in filename:
 
		if ch not in string.ascii_letters+string.digits+"/":
 
			escaped += "\\"
 
		escaped += ch
 
	return escaped
 
 
def shares(config_path):
 
	if not os.path.exists(config_path):
 
		print "Need config file to build shares, creating..."
 
		writeConfig(config_path)
 
	r = os.popen('muscan -c %s -l' % escapeCommand(config_path)).read()
 
	_shared = [i for i in r.split("\n") if i]
 
	print
 
	print
 
	print 'Shared directories setup'
 
	print ' Note: these options work immediately and without confirmation'
 
	print
 
	print '1. Rescan shares incrementally'
 
	print '2. Rescan shares from scratch'
 
	print
 
	print '3. Add shared directory'
 
	print
 
	ix = 3
 
	for dir in _shared:
 
		ix = ix + 1
 
		print '%i. Remove %s' % (ix, dir)
 
	print
 
	print '0. Leave this menu'
 
	c = choice(ix)
 
	if c == 0:
 
		return 1
 
	elif c == 1:
 
		print
 
		print 'Please wait patiently while rescanning shares...'
 
		os.system('muscan -c %s' % escapeCommand(config_path))
 
	elif c == 2:
 
		print
 
		print 'Please wait patiently while rescanning shares...'
 
		os.system('muscan -c %s -r' % escapeCommand(config_path))
 
	elif c == 3:
 
		dir = string_choice('Path', '')
 
		if dir:
 
			dir = os.path.expanduser(dir)
 
			if dir[-1:] == "/":
 
				dir = dir[:-1]
 
			print dir 
 
			print 'Please wait patiently while rescanning shares...'
 
			os.system('muscan -c %s -s %s' % (escapeCommand(config_path), escapeCommand(dir)))
 
	else:
 
		os.system('muscan -c %s -u %s' % (escapeCommand(config_path), escapeCommand(_shared[c - 4])))
 
	
 
	return 0
 
 
def bshares(config_path):
 
	if not os.path.exists(config_path):
 
		print "Need config file to build shares, creating..."
 
		writeConfig(config_path)
 
	r = os.popen('muscan -c %s -b -l' % escapeCommand(config_path)).read()
 
	_shared = [i for i in r.split("\n") if i]
 
	_have_buddy_shares = config['transfers'].get('have_buddy_shares', '')
 
	print
 
	print
 
	print 'Special Buddy-only Shared directories setup'
 
	print ' Note: these options work immediately and without confirmation'
 
	print
 
	print '1. Rescan shares incrementally'
 
	print '2. Rescan shares from scratch'
 
	print
 
	print '3. Special list for buddies:', _have_buddy_shares
 
	print
 
	print '4. Add shared directory'
 
	print
 
	ix = 4
 
	for dir in _shared:
 
		ix = ix + 1
 
		print '%i. Remove %s' % (ix, dir)
 
	print
 
	print '0. Leave this menu'
 
	c = choice(ix)
 
	if c == 0:
 
		return 1
 
	elif c == 1:
 
		print
 
		print 'Please wait patiently while rescanning shares...'
 
		os.system('muscan -c %s -b' % escapeCommand(config_path))
 
	elif c == 2:
 
		print
 
		print 'Please wait patiently while rescanning shares...'
 
		os.system('muscan -c %s -b -r' % escapeCommand(config_path))
 
	elif c == 3:
 
		while not buddyshares(): pass
 
	elif c == 4:
 
		dir = string_choice('Path', '')
 
		if dir:
 
			dir = os.path.expanduser(dir)
 
			if dir[-1:] == "/":
 
				dir = dir[:-1]
 
			print dir 
 
			print 'Please wait patiently while rescanning shares...'
 
			os.system('muscan -c %s -b -s %s' % (escapeCommand(config_path), escapeCommand(dir)))
 
	else:
 
		os.system('muscan -c %s -b -u %s' % (escapeCommand(config_path), escapeCommand(_shared[c - 5])))
 
	
 
	return 0
 
	
 
 
def save(config_path):
 
	print
 
	print
 
	print 'Leave museek setup'
 
	print
 
	print '1. Save changes'
 
	print '2. Don\'t save changes'
 
	print
 
	print '0. Don\'t leave museek setup'
 
	c = choice(2)
 
	if c == -1:
 
		return 0
 
	elif c == 0:
 
		return 1
 
	elif c == 1:
 
		writeConfig(config_path)
 
	return 2
 
	
 
def main(config_path):
 
	
 
	print
 
	print 'Museek daemon setup'
 
	print
 
	print "Settings that must be set for museekd to run: \nServer, Username, Password, Interface Password, Download Dir"
 
	print
 
	print '1. Soulseek server'
 
	print '2. Peer connections'
 
	print '3. Interface listeners'
 
	print '4. Network and filesystem character sets'
 
	print '5. Download and incomplete file directory'
 
	print
 
	print '6. Shared directories'
 
	print "7. Buddy-Only Shared directories"
 
	print
 
	print '0. Exit museek setup'
 
	c = choice(7)
 
	if c == -1:
 
		return 0
 
	elif c == 1:
 
		while not server(): pass
 
	elif c == 2:
 
		while not portrange(): pass
 
	elif c == 3:
 
		while not ifaces(): pass
 
	elif c == 4:
 
		while not charsets(): pass
 
	elif c == 5:
 
		while not dirs(): pass
 
	elif c == 6:
 
		while not shares(config_path): pass
 
	elif c == 7:
 
		while not bshares(config_path): pass
 
	else:
 
		while 1:
 
			c = save(config_path)
 
			if c == 1:
 
				return 0
 
			elif c == 2:
 
				return  1
 
	return 0
 
 
if '--help' in sys.argv or len(sys.argv) > 2:
 
	print
 
	print 'Syntax: %s [path]' % sys.argv[0]
 
	print
 
	print '\'path\' is the filename of the configuration file to edit.'
 
	print
 
	sys.exit(1)
 
 
if len(sys.argv) == 2:
 
	path = sys.argv[1]
 
else:
 
	path = DEFAULT_CONFIG
 
 
path = os.path.expanduser(path)
 
try:
 
	readConfig(path)
 
except:
 
	print
 
	print "Warning: couldn't load configuration, loading template"
 
	print
 
	readTemplate(DEFAULT_TEMPLATE, path)
 
 
try:
 
	while not main(path): pass
 
except Exception, e:
 
	
 
	print "\nAborting musetup..."			 		 |