fish-shell/web_config/webconfig.py

176 lines
4.6 KiB
Python
Raw Normal View History

2012-03-15 18:43:45 +08:00
#!/usr/bin/python
import SimpleHTTPServer
import SocketServer
import webbrowser
import subprocess
2012-03-16 18:03:43 +08:00
import re, json, socket, sys
2012-03-15 18:43:45 +08:00
def run_fish_cmd(text):
from subprocess import PIPE
p = subprocess.Popen(["fish"], stdin=PIPE, stdout=PIPE, stderr=PIPE)
out, err = p.communicate(text)
return out, err
2012-03-23 01:23:07 +08:00
named_colors = {
'black' : '000000',
'red' : 'FF0000',
'green' : '00FF00',
'brown' : '725000',
'yellow' : 'FFFF00',
'blue' : '0000FF',
'magenta' : 'FF00FF',
'purple' : 'FF00FF',
'cyan' : '00FFFF',
'white' : 'FFFFFF'
}
def parse_color(color_str):
""" A basic function to parse a color string, for example, 'red' '--bold' """
comps = color_str.split(' ')
print "comps: ", comps
color = 'normal'
bold, underline = False, False
for comp in comps:
# Remove quotes
comp = comp.strip("'\" ")
if comp == '--bold':
bold = True
elif comp == '--underline':
underline = True
elif comp in named_colors:
# Named color
color = named_colors[comp]
elif re.match(r"[0-9a-fA-F]{3}", comp) is not None or re.match(r"[0-9a-fA-F]{6}", comp) is not None:
# Hex color
color = comp
else:
# Unknown component
pass
return color
2012-03-17 08:21:37 +08:00
class FishVar:
""" A class that represents a variable """
def __init__(self, name, value):
self.name = name
self.value = value
self.universal = False
self.exported = False
def get_json_obj(self):
# Return an array(3): name, value, flags
flags = []
if self.universal: flags.append('universal')
if self.exported: flags.append('exported')
return [self.name, self.value, ', '.join(flags)]
2012-03-15 18:43:45 +08:00
class FishConfigHTTPRequestHandler(SimpleHTTPServer.SimpleHTTPRequestHandler):
def do_get_colors(self):
"Look for fish_color_*"
result = []
2012-03-23 01:23:07 +08:00
out, err = run_fish_cmd('set -L')
for line in out.split('\n'):
for match in re.finditer(r"^fish_color_(\S+) (.+)", line):
color_name, color_value = match.group(1, 2)
result.append([color_name.strip(), parse_color(color_value)])
2012-03-15 18:43:45 +08:00
print result
return result
2012-03-16 18:03:43 +08:00
def do_get_functions(self):
out, err = run_fish_cmd('functions')
out = out.strip()
2012-03-23 01:23:07 +08:00
print out
2012-03-16 18:03:43 +08:00
# Not sure why fish sometimes returns this with newlines
if "\n" in out:
return out.split('\n')
else:
return out.strip().split(', ')
2012-03-17 08:21:37 +08:00
def do_get_variable_names(self, cmd):
" Given a command like 'set -U' return all the variable names "
out, err = run_fish_cmd(cmd)
return out.split('\n')
def do_get_variables(self):
out, err = run_fish_cmd('set -L')
2012-03-17 08:21:37 +08:00
# Put all the variables into a dictionary
vars = {}
for line in out.split('\n'):
comps = line.split(' ', 1)
if len(comps) < 2: continue
fish_var = FishVar(comps[0], comps[1])
vars[fish_var.name] = fish_var
# Mark universal variables. L means don't abbreviate.
for name in self.do_get_variable_names('set -nUL'):
2012-03-17 08:21:37 +08:00
if name in vars: vars[name].universal = True
# Mark exported variables. L means don't abbreviate.
for name in self.do_get_variable_names('set -nxL'):
2012-03-17 08:21:37 +08:00
if name in vars: vars[name].exported = True
return [vars[key].get_json_obj() for key in sorted(vars.keys(), key=str.lower)]
def do_get_history(self):
# Use \x1e ("record separator") to distinguish between history items. The first
# backslash is so Python passes one backslash to fish
out, err = run_fish_cmd('for val in $history; echo -n $val \\x1e; end')
return out.split('\x1e')
2012-03-16 18:03:43 +08:00
2012-03-15 18:43:45 +08:00
2012-03-17 08:21:37 +08:00
def do_get_color_for_variable(self, name):
"Return the color with the given name, or the empty string if there is none"
out, err = run_fish_cmd("echo -n $" + name)
return out
2012-03-15 18:43:45 +08:00
def do_GET(self):
p = self.path
if p == '/colors/':
output = self.do_get_colors()
2012-03-16 18:03:43 +08:00
elif p == '/functions/':
output = self.do_get_functions()
2012-03-17 08:21:37 +08:00
elif p == '/variables/':
output = self.do_get_variables()
elif p == '/history/':
output = self.do_get_history()
2012-03-17 08:21:37 +08:00
elif re.match(r"/color/(\w+)/", p):
name = re.match(r"/color/(\w+)/", p).group(1)
output = self.do_get_color_for_variable(name)
2012-03-15 18:43:45 +08:00
else:
return SimpleHTTPServer.SimpleHTTPRequestHandler.do_GET(self)
# Return valid output
self.send_response(200)
self.send_header('Content-type','text/html')
self.wfile.write('\n')
# Output JSON
json.dump(output, self.wfile)
2012-03-16 18:03:43 +08:00
PORT = 8000
while PORT <= 9000:
try:
Handler = FishConfigHTTPRequestHandler
httpd = SocketServer.TCPServer(("", PORT), Handler)
# Success
break;
except socket.error:
type, value = sys.exc_info()[:2]
if 'Address already in use' not in value:
break
PORT += 1
if PORT > 9000:
print "Unable to start a web server"
sys.exit(-1)
2012-03-15 18:43:45 +08:00
2012-03-16 18:03:43 +08:00
2012-03-15 18:43:45 +08:00
webbrowser.open("http://localhost:%d" % PORT)
print "serving at port", PORT
httpd.serve_forever()