# HG changeset patch
# User Malte Bayer <mbayer@neo-soft.org>
# Date 1354472363 -3600
# Node ID bb150048d32936e6edfe5c2a22a0d9fa12092f3a
# Parent  313aad7435a893ced07fa3d26bad43732d770e10
added player profiles

diff -r 313aad7435a8 -r bb150048d329 slotUI/SlotCli.py
--- a/slotUI/SlotCli.py	Sun Dec 02 18:03:06 2012 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,651 +0,0 @@
-#!/usr/bin/env python
-"""
-Freeslot project
-Command line interface
-"""
-
-from freeslot import Blackbox, LOGLEVEL
-from optparse import OptionParser
-from operator import itemgetter
-from subprocess import Popen, PIPE
-import sys, os
-from copy import copy
-import curses
-from time import sleep
-
-import SimpleXMLRPCServer
-import xmlrpclib
-import threading
-
-
-VERSION = "1.7"
-MAXSLOTS = 6
-TERM = {
-    "caption": "\033[1;37m\033[1;44m",
-    "text": "\033[1;30m",
-    }
-
-# disable debug log output
-LOGLEVEL = 10
-
-SOUNDPREFIX = "quake-"
-EVENTPREFIX = "event/"
-SOUNDS = {
-        "countdown_start": os.path.abspath(SOUNDPREFIX + "sound/countdown.mp3"),
-        "race_start":      os.path.abspath(SOUNDPREFIX + "sound/racestart.mp3"),
-        "race_prepare":    os.path.abspath(SOUNDPREFIX + "sound/prepare.mp3"),
-        "lap_record":      os.path.abspath(SOUNDPREFIX + "sound/laprecord.mp3"),
-        "fuel_warning1":   os.path.abspath(SOUNDPREFIX + "sound/fuel1.mp3"),
-        "fuel_warning2":   os.path.abspath(SOUNDPREFIX + "sound/fuel2.mp3"),
-        "fuel_full":       os.path.abspath(SOUNDPREFIX + "sound/fuel_full.mp3"),
-        "pitlane_enter":   os.path.abspath(SOUNDPREFIX + "sound/pitlane_enter.mp3"),
-        "pitlane_exit":    os.path.abspath(SOUNDPREFIX + "sound/pitlane_exit.mp3"),
-        "data_error":      os.path.abspath(SOUNDPREFIX + "sound/data_error.mp3"),
-        "panic":           os.path.abspath(SOUNDPREFIX + "sound/panic.mp3"),
-        "panic_shortcut":  os.path.abspath(SOUNDPREFIX + "sound/panic_shortcut.mp3"),
-        "resume":          os.path.abspath(SOUNDPREFIX + "sound/resume.mp3"),
-        "win":             os.path.abspath(SOUNDPREFIX + "sound/win.mp3"),
-
-    }
-
-def trigger_sound(what):
-    if what in SOUNDS:
-        Popen(["/usr/bin/mpg123", "-q", SOUNDS[what]])
-        #os.spawnlp(os.P_NOWAIT, "/usr/bin/mpg123", "mpg123", SOUNDS[what])
-        #Popen(["/usr/bin/mpg123", SOUNDS[what]]).pid
-
-def trigger_event(what, slot = 0):
-    trigger_sound(what)
-    Popen(["/bin/sh", os.path.abspath(EVENTPREFIX + what), str(slot)])
-
-class SlotServer(threading.Thread):
-    def __init__(self, blackbox):
-        threading.Thread.__init__(self)
-        self.server = SimpleXMLRPCServer.SimpleXMLRPCServer(("localhost", 8000))
-        self.server.register_instance(blackbox)
-        #self.server.register_function(lambda astr: '_' + astr, '_string')
-        self.daemon = True
-
-    def run(self):
-        self.server.serve_forever()
-
-class SlotClient():
-    def __init__(self, url):
-        self.box = xmlrpclib.Server(url)
-
-class SlotCli():
-    def __init__(self, test = None, dev=""):
-        self.box = Blackbox()
-        if (not test):
-            self.box.connect(dev)
-            self.rpcserver = SlotServer(self.box)
-            self.rpcserver.start()
-        self.slot_dummy = {
-            "name": "Unnamed",
-            "laps": 0,
-            "laps_last": 0,
-            "last": 0.00,
-            "best": 0.00,
-            "fuel": 0,
-            "fuel_last": 0,
-            "position": 0,
-            "drive": 0,
-            "status": "Idle",
-            "clk": 0,
-            "car": 0,
-            "limit": 15,
-            }
-
-        self.slot = [
-            copy(self.slot_dummy), copy(self.slot_dummy),
-            copy(self.slot_dummy), copy(self.slot_dummy),
-            copy(self.slot_dummy), copy(self.slot_dummy),
-            ]
-        self.reset_slots()
-        self.sysclk = 0.00
-        self.sysclk_last = 0.00
-        self.bestlap = 9999999.00
-        self.test = test
-        self.laplimit = 999;
-        self.timelimit = 0;
-
-    def reset_slots(self):
-        idx = 0
-        for slt in self.slot:
-            slt["laps"] = 0
-            slt["laps_last"] = 0
-            slt["last"] = 0.00
-            slt["best"] = 0.00
-            slt["fuel"] = 100
-            slt["fuel_last"] = 0
-            slt["position"] = idx
-            slt["car"] = idx # used for sort order calculation
-            slt["status"] = self.slot_dummy["status"]
-            slt["clk"] = 0
-            slt["limit"] = 15
-            idx += 1
-        self.bestlap = 99999.00
-        self.raceactive = False
-
-    def update_positions(self):
-        order1 = sorted(self.slot, key=itemgetter(
-            "clk"))
-        order2 = sorted(self.slot, key=itemgetter(
-            "laps"), reverse=True)
-        idx = 1
-        for tst in order2:
-            self.slot[tst["car"]]["position"] = idx
-            idx += 1
-
-    def render_slots(self):
-        self.update_positions()
-        self.scr.addstr(3,0,
-            #"Pos | #/Name            | Laps | Best     | Last     | Fuel | Status    ",
-            "Pos | #/Name                                                            ",
-            curses.color_pair(2))
-        self.scr.addstr(4,4,
-            "  Laps | Best     | Last     | Fuel | Status                        ",
-            curses.color_pair(2))
-        for idx in range(MAXSLOTS):
-            """
-            self.scr.addstr((3 + (self.slot[idx]["position"] * 2)), 0,
-                "%3i | %i %15s | %4i | %7.2fs | %7.2fs | %3i%% | %10s" % (
-                self.slot[idx]["position"],
-                self.slot[idx]["car"] + 1, self.slot[idx]["name"],
-                self.slot[idx]["laps"],
-                self.slot[idx]["best"],
-                self.slot[idx]["last"],
-                self.slot[idx]["fuel"],
-                self.slot[idx]["status"],
-                ),
-                curses.color_pair(11 + idx) )
-            """
-            if idx > 3:
-                namesuffix = " (%i)" % self.slot[idx]["drive"]
-            else:
-                namesuffix = ""
-
-            self.scr.addstr((3 + (self.slot[idx]["position"] * 2)), 0,
-                "%3i | %i %15s %48s" % (
-                self.slot[idx]["position"],
-                self.slot[idx]["car"] + 1, (self.slot[idx]["name"] + namesuffix),
-                "",
-                ),
-                curses.color_pair(11 + idx) )
-            self.scr.addstr((4 + (self.slot[idx]["position"] * 2)), 4,
-                "  %4i | %7.2fs | %7.2fs | %3i%% | %10s | %2i% 15s" % (
-                self.slot[idx]["laps"],
-                self.slot[idx]["best"],
-                self.slot[idx]["last"],
-                self.slot[idx]["fuel"],
-                self.slot[idx]["status"],
-                self.slot[idx]["limit"],
-                ""
-                ),
-                curses.color_pair(11 + idx) )
-
-    def cleartop(self):
-        self.scr.addstr(0,0, "%60s" % "Race Limits: %i Laps / %i Minutes" % (self.laplimit, self.timelimit))
-        self.scr.addstr(1,0, "%80s" % " ")
-
-    def readName(self, slot):
-        self.scr.nodelay(0) # enable delay on readkey
-        curses.echo()
-        self.scr.addstr(0,0, "Enter Name for Controller %i [%s]:" % (
-            slot + 1,
-            self.slot[slot]["name"]),
-            curses.color_pair(1))
-        self.scr.refresh()
-        name = self.scr.getstr(1,0, 15)
-        if name != "":
-            self.slot[slot]["name"] = name
-        self.cleartop()
-        self.scr.refresh()
-        curses.noecho()
-        self.scr.nodelay(1) # disable delay on readkey
-
-    def readLimit(self, slot):
-        limit = self.readInt("SPEEDLIMIT for %s (%i)" % (
-            self.slot[slot]["name"],
-            slot + 1),
-            self.slot[slot]["limit"], 15)
-        if limit:
-            self.slot[slot]["limit"] = limit
-            self.cleartop()
-            self.box.speedlimit(slot, limit)
-
-
-    def readInt(self, msg, default, maximum = 999999):
-        self.scr.nodelay(0) # enable delay on readkey
-        curses.echo()
-        self.scr.addstr(0,0, "%s [%i]:" % (
-            msg,
-            default),
-            curses.color_pair(1))
-        self.scr.refresh()
-        inp = self.scr.getstr(1,0, 4)
-        if inp != "":
-            try:
-                inp = int(inp)
-                if inp > maximum: inp = maximum
-            except Exception:
-                inp = None
-        else:
-            inp = None
-        self.cleartop()
-        self.scr.refresh()
-        curses.noecho()
-        self.scr.nodelay(1) # disable delay on readkey
-        return inp
-
-
-    def monitor_init(self, live = 1):
-        """
-        Send initializing commands for live monitoring
-        """
-        self.box.query("F1\n") # set fuel logic enabled
-        self.box.query("*%i\n" % live) # set live fuel info
-
-    def monitor_learn(self, slot):
-        # clear garbage in UART rx buffer
-        self.box.query("*0\n") # set live fuel info
-        self.box.query("*0\n") # set live fuel info
-        while self.box.readline() != "": pass
-
-        trk = False
-        spd = 0
-        trk_old = False
-        spd_old = 0
-        clock = -1
-
-        self.monitor_init(slot + 2)
-        while 1:
-            #key = self.scr.getch()
-            #if key == ord('c'): break
-
-            # is there something in the rx buffer?
-            rx = self.box.readline()
-            if (rx != ""):
-                try:
-                    data = rx.split(":")
-                    if rx[:3] == "LN:":
-                        if clock >= 0:
-                            clock += 1
-                        spd = int(data[1], 16)
-                        trk = (data[2] != 'X')
-                        if (spd != spd_old) or (trk != trk_old):
-                            if clock < 0:
-                                clock = 0
-                            print "%i,%i,%s" % (clock, spd, trk)
-                        trk_old = trk
-                        spd_old = spd * 1
-                    if rx[:2] == "L:":
-                        # update lap time info
-                        l = int(data[2], 16)
-                        s = int(data[3]) - 1
-                        t = int(data[4], 16) / 2000.00
-                        if (slot == s):
-                            print "# lap %i complete: %3.2f seconds" % (l, t)
-                            clock = 0
-                            print "%i,%i,%s" % (clock, spd, trk)
-                except:
-                    print "RX ERROR: " % rx
-
-    def monitor_playback(self, slot, filename):
-        # clear garbage in UART rx buffer
-        self.box.query("*0\n") # set live fuel info
-        self.box.query("*0\n") # set live fuel info
-        sleep(1)
-        cli.box.speedminimum(slot, 0 )
-        while self.box.readline() != "": pass
-
-        clock = -5
-        trkfile = open(filename, "r").readlines()
-        print "Loading %s..." % filename
-
-        while 1:
-            try:
-                for l in trkfile:
-                    l = l.strip()
-                    if (l != "") and (l[:1] != "#"):
-                        print "Line: %s" % repr(l)
-                        data = l.split(",")
-                        speed = int(data[1])
-                        while (clock < int(data[0]) and (int(data[0]) > 0)):
-                            clock += 1
-                            sleep(0.07)
-                        print "CLK %i/%i -> set: %i" % (clock, int(data[0]), speed)
-                        cli.box.speedminimum(slot, speed )
-                # now wait for lap sync :)
-                while self.box.readline() != "": pass
-                rx = ""
-                while rx[:2] != "L:":
-                    rx = self.box.readline()
-                data = rx.split(":")
-                l = int(data[2], 16)
-                s = int(data[3]) - 1
-                t = int(data[4], 16) / 2000.00
-                print "# lap %i complete: %3.2f seconds" % (l, t)
-                clock = -3
-            except Exception, e:
-                print repr(e)
-                sys.exit(1)
-            except KeyboardInterrupt:
-                print "resetting"
-                cli.box.speedminimum(slot, 0 )
-                cli.box.speedminimum(slot, 0 )
-                sys.exit(0)
-                
-
-
-    def monitor(self):
-        """
-        Live Monitor on the console
-        Keyboard loop to control it???
-        """
-        # clear garbage in UART rx buffer
-        while self.box.readline() != "": pass
-
-        self.monitor_init()
-        self.scr = curses.initscr()
-        curses.start_color()
-        curses.init_pair(1, curses.COLOR_WHITE, curses.COLOR_BLACK) # standard text
-        curses.init_pair(2, curses.COLOR_WHITE, curses.COLOR_BLUE) # label
-        curses.init_pair(11, curses.COLOR_BLACK, curses.COLOR_YELLOW) # player 1 slot
-        curses.init_pair(12, curses.COLOR_BLACK, curses.COLOR_GREEN) # player 2 slot
-        curses.init_pair(13, curses.COLOR_BLACK, curses.COLOR_RED) # player 3 slot
-        curses.init_pair(14, curses.COLOR_BLACK, curses.COLOR_MAGENTA) # player 4 slot
-        curses.init_pair(15, curses.COLOR_WHITE, curses.COLOR_BLACK) # player 5 slot
-        curses.init_pair(16, curses.COLOR_WHITE, curses.COLOR_BLACK) # player 6 slot
-        curses.noecho() # disable key echo
-        curses.cbreak() # do not buffer keypresses
-        self.scr.keypad(1) # enable special keys
-        self.scr.nodelay(1) # disable delay on readkey
-
-        self.cleartop()
-        self.render_slots()
-        self.scr.refresh()
-
-
-        while 1:
-            key = self.scr.getch()
-            if key == ord('c'): break
-            elif key == ord(' '): self.box.query("+") # panic / resume
-            elif key == 10: self.box.query("#") # remote start button press
-            elif key == ord('1'): self.readName(0)
-            elif key == ord('2'): self.readName(1)
-            elif key == ord('3'): self.readName(2)
-            elif key == ord('4'): self.readName(3)
-            elif key == ord('5'): self.readName(4)
-            elif key == ord('6'): self.readName(5)
-            elif key == ord('q'): self.readLimit(0)
-            elif key == ord('w'): self.readLimit(1)
-            elif key == ord('e'): self.readLimit(2)
-            elif key == ord('r'): self.readLimit(3)
-            elif key == ord('t'): self.readLimit(4)
-            elif key == ord('z'): self.readLimit(5)
-            elif key == ord('a'):
-                if self.slot[4]["drive"] > 0: self.slot[4]["drive"] -= 1
-                cli.box.speedminimum(4, self.slot[4]["drive"])
-            elif key == ord('s'):
-                if self.slot[4]["drive"] < 16: self.slot[4]["drive"] += 1
-                cli.box.speedminimum(4, self.slot[4]["drive"])
-            elif key == ord('y'):
-                if self.slot[5]["drive"] > 0: self.slot[5]["drive"] -= 1
-                cli.box.speedminimum(5, self.slot[5]["drive"])
-            elif key == ord('x'):
-                if self.slot[5]["drive"] < 16: self.slot[5]["drive"] += 1
-                cli.box.speedminimum(5, self.slot[5]["drive"])
-            elif key == ord('t'):
-                tmp = self.readInt("Set new Race TIME limit", self.timelimit)
-                if tmp: self.timelimit = tmp
-            elif key == ord('l'):
-                tmp = self.readInt("Set new Race LAP limit", self.laplimit)
-                if tmp: self.laplimit = tmp
-
-
-            # is there something in the rx buffer?
-            rx = self.box.readline()
-            if (rx != "") or self.test:
-                self.scr.addstr(17,0,
-                    "Last RX: %19s" % rx, curses.color_pair(2))
-                self.scr.redrawwin()
-                self.scr.refresh()
-                # we have received something
-                try:
-                    data = rx.split(":")
-                    if rx[:2] == "L:":
-                        # update lap time info
-                        l = int(data[2], 16)
-                        slot = int(data[3]) - 1
-                        t = int(data[4], 16) / 2000.00
-                        self.sysclk = int(data[5], 16) / 2000.00
-                        self.slot[slot]["laps_last"] = self.slot[slot]["laps"]
-                        self.slot[slot]["laps"] = l
-                        self.slot[slot]["last"] = t
-                        self.slot[slot]["clk"] = self.sysclk
-                        if (self.slot[slot]["best"] > t) or (self.slot[slot]["best"] == 0):
-                            self.slot[slot]["best"] = t
-                        if self.bestlap > t:
-                            trigger_event("lap_record", slot + 1)
-                            self.bestlap = t
-
-                        self.slot[slot]["status"] = "IN-RACE"
-                        if (self.slot[slot]["laps_last"] != l) and (l == self.laplimit):
-                            # we have lap limit reached!
-                            trigger_event("win", slot + 1)
-                            self.raceactive = False
-                            self.slot[slot]["status"] = "WINNER!"
-                            self.box.query("+") # stop race
-
-                        self.render_slots()
-
-                    if rx[:2] == "F:":
-                        # update fuel level
-                        slot = int(data[1])
-                        f = int(data[2], 16)
-                        f = f / 100 # fuel in percent
-                        self.sysclk = int(data[3], 16) / 2000.00
-                        self.slot[slot]["fuel_last"] = self.slot[slot]["fuel"]
-                        self.slot[slot]["fuel"] = f
-                        if self.slot[slot]["fuel_last"] != f:
-                            if (self.slot[slot]["fuel_last"] == 16) and (f == 15):
-                                # 15 percent fuel, set speed limit for car to 8
-                                # warning sound
-                                trigger_event("fuel_warning1", slot + 1)
-                                cli.box.speedlimit(slot, 8)
-                            if (self.slot[slot]["fuel_last"] == 6) and (f == 5):
-                                # 5 percent, set speed limit for car to 6
-                                # warning sound
-                                trigger_event("fuel_warning2", slot + 1)
-                                cli.box.speedlimit(slot, 6)
-                            if (self.slot[slot]["fuel_last"] == 1) and (f == 0):
-                                # fuel empty
-                                # set speedlimit to 4
-                                cli.box.speedlimit(slot, 4)
-                            if (self.slot[slot]["fuel_last"] < f) and (f >= 11) and (f < 20):
-                                cli.box.speedlimit(slot, 15)
-                            if (self.slot[slot]["fuel_last"] < f) and (f == 100):
-                                trigger_event("fuel_full", slot + 1)
-                        self.render_slots()
-
-                    if rx[:1] == "~":
-                        # jumpstart occured
-                        slot = int(rx[1:2])
-                        t = int(data[1], 16) / 2000.00
-                        self.slot[slot]["jumpstart"] = t
-                        self.slot[slot]["status"] = "Jumpstart!"
-
-                    if rx[:3] == "RW:":
-                        # ResponseWire packet, do nothing at the moment, just decode
-                        slot = int(data[1])
-                        devtype = int(data[2])
-                        sender = int(data[3], 16)
-                        status = int(data[4], 16)
-                        self.sysclk = int(data[5], 16)
-                        if (devtype == 4):
-                            # pitlane sent something
-                            if (status == 5):
-                                self.slot[slot]["status"] = "PITLANE"
-                                trigger_event("pitlane_enter", slot + 1)
-                            if (status == 7):
-                                self.slot[slot]["status"] = "IN-RACE"
-                                trigger_event("pitlane_exit", slot + 1)
-
-                        self.render_slots()
-
-                    if rx == "!RACE PREPARE":
-                        # reset current race status
-                        # and display PREPARE PHASE
-                        self.reset_slots()
-                        for slot in range(MAXSLOTS):
-                            self.slot[slot]["status"] = "Prepare"
-                        trigger_event("race_prepare")
-
-                    if rx == "!RACE START":
-                        for slot in range(MAXSLOTS):
-                            if self.slot[slot]["status"] == "~~~~~~~~~~":
-                                self.slot[slot]["status"] = "Idle"
-                        trigger_event("race_start")
-                        self.raceactive = True
-
-                    if rx == "!COUNTDOWN":
-                        # countdown initiated
-                        for slot in range(MAXSLOTS):
-                            self.slot[slot]["status"] = "~~~~~~~~~~"
-                        trigger_event("countdown_start")
-
-                    if rx == "!PANIC":
-                        # panic mode
-                        trigger_event("panic")
-
-                    if rx == "!SHORTCUT":
-                        # panic mode
-                        trigger_event("panic_shortcut")
-
-                    if rx == "!RESUME":
-                        # panic mode
-                        trigger_event("resume")
-
-
-                    if ((self.timelimit > 0) and (self.raceactive) and 
-                        (self.sysclk_last != self.sysclk) and 
-                        ((self.sysclk / 60) >= self.timelimit)):
-                        self.sysclk_last = self.sysclk
-                        self.raceactive = False
-                        # we have time limit reached!
-                        self.box.query("+") # stop race
-                        trigger_event("win")
-                        # get the one with position 1
-                        for slot in self.slots:
-                            if slot["position"] == 1:
-                                slot["status"] = "WINNER!"
-                	self.render_slots()
-                    self.sysclk_last = self.sysclk
-    
-
-
-                    self.scr.addstr(17,31,
-                        "Race Timer: %7.3f min" % (self.sysclk / 60),
-                        curses.color_pair(2))
-
-                    self.scr.refresh()
-
-                except Exception:
-                    trigger_event("data_error")
-                    pass
-
-        # terminate
-        curses.nocbreak()
-        self.scr.keypad(0)
-        curses.echo()
-        curses.endwin()
-        return None
-
-    def cyclemode(self):
-        pass
-
-if __name__ == "__main__":
-    parser = OptionParser(version="%prog " + VERSION)
-    parser.add_option("--live", dest="live", action="store_true", default=False,
-        help="Run Live monitor on console", metavar="[0-5]")
-    parser.add_option("--learn", dest="learn", action="store_true", default=False,
-        help="Run Learning mode for [slot]", metavar="[0-5]")
-    parser.add_option("--teach", dest="playback", 
-        help="Playback teach file", metavar="[filename]")
-
-    parser.add_option("--slot", dest="carid",
-        help="Required for programming a car directly", metavar="[1-6]")
-    parser.add_option("--fuel", dest="fuel",
-        help="Set maximum CAR fuel level", metavar="[0-15]")
-    parser.add_option("--brake", dest="brake",
-        help="Set CAR brake strength", metavar="[0-15]")
-    parser.add_option("--accel", dest="accel",
-        help="Set CAR acceleration ", metavar="[6-15]")
-    parser.add_option("--blink", dest="blink",
-        help="Set car lights blinking state", metavar="[on|off]")
-    parser.add_option("--limit", dest="limit",
-        help="Controlled SPEED LIMIT (15 = no limit)", metavar="[0-15]")
-    parser.add_option("--drive", dest="drive",
-        help="Controlled SPEED MINIMUM (0 = disabled)", metavar="[0-15]")
-    parser.add_option("--test", dest="test", action="store_true", default=False,
-        help="", metavar="")
-    parser.add_option("--dev", dest="dev", default="/dev/ttyUSB0",
-        help="Communication port", metavar="[/dev/ttyUSB0]")
-
-    (options, args) = parser.parse_args()
-    #if not options.dev:
-    #    options.dev = "/dev/ttyUSB0"
-
-    if options.live or options.learn or options.playback:
-        cli = SlotCli(options.test, options.dev)
-    else:
-        cli = SlotClient('http://localhost:8000')
-    # should a CLI function be started?
-
-    if options.live:
-        # start the live monitor
-        cli.monitor()
-        sys.exit(0)
-
-    # check commandline if we have to program something
-    if not options.carid:
-        print "Option --slot is required for all car programming commands!\nUse --help to get a list of available commands"
-        sys.exit(1)
-    else:
-        options.carid = int(options.carid) - 1
-        if (options.carid < 0) or (options.carid > 6):
-            print "Error: Invalid slot selected"
-            sys.exit(1)
-
-    if options.learn:
-        # start the learn monitor
-        cli.monitor_learn(options.carid)
-        sys.exit(0)
-
-    if options.playback:
-        # start the playback monitor
-        cli.monitor_playback(options.carid, options.playback)
-        sys.exit(0)
-
-    if options.fuel:
-        print "setFuel: " + cli.box.progcar(int(options.carid), "fuel", int(options.fuel))
-
-    if options.accel:
-        print "setAccel: " + cli.box.progcar(int(options.carid), "accel", int(options.accel))
-
-    if options.brake:
-        print "setBrake: " + cli.box.progcar(int(options.carid), "brake", int(options.brake))
-
-    if options.blink:
-        state = False
-        if options.blink == "on":
-            state = True
-        print "setBlink: " + cli.box.blinkcar(int(options.carid), state)
-
-    if options.limit:
-        print "Change Speed Limit: " + cli.box.speedlimit(int(options.carid), int(options.limit))
-
-    if options.drive:
-        print "Change minimum Speed drive: " + cli.box.speedminimum(int(options.carid), int(options.drive))
-
diff -r 313aad7435a8 -r bb150048d329 slotUI/SlotUi.py
--- a/slotUI/SlotUi.py	Sun Dec 02 18:03:06 2012 +0100
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,80 +0,0 @@
-#!/usr/bin/env python
-
-import pygtk
-pygtk.require('2.0')
-import gtk
-
-class Window:
-    """
-    Base Window Class
-    """
-    def delete_event(self, widget, event, data=None):
-        return False
-
-    def __init__(self, title="unnamed window"):
-        self.window = gtk.Window(gtk.WINDOW_TOPLEVEL)
-        self.window.connect("delete_event", self.delete_event)
-        self.window.set_title(title)
-
-class SlotCars(Window):
-    """
-    Car configuration window
-    """
-    def __init__(self):
-        Window.__init__(self, "Car configuration")
-        #self.slot = gtk.ComboBox("Select car")
-        #self.slot.show()
-        #window.add(self.slot)
-
-class SlotUi(Window):
-    """
-    Graphical User Interface
-    using GTK
-    """
-    def delete_event(self, widget, event, data=None):
-        if widget == self.window:
-            self.destroy(widget, data)
-        Window.delete_event(self, widget, event, data)
-        return False
-
-    def destroy(self, widget, data=None):
-        gtk.main_quit()
-
-    def __init__(self):
-        Window.__init__(self, "FreeSlot UI")
-
-        # define toolbar buttons
-        self.buttons = {
-                "config":       gtk.Button("Settings"),
-                "cars":         gtk.Button("Car Config"),
-            }
-        self.buttons["config"].connect("clicked", self.openwindow, "config")
-        self.buttons["cars"].connect("clicked", self.openwindow, "cars")
-        #setup the toolbar box
-        self.toolbar = gtk.HBox(False, 0)
-        self.toolbar.show()
-        for btn in self.buttons:
-            self.toolbar.add(self.buttons[btn])
-            self.buttons[btn].show()
-
-        # create subwindow objects
-        self.cars = SlotCars()
-        self.windows = {
-            "config": None,
-            "cars": self.cars.window,
-            }
-
-        self.window.add(self.toolbar)
-        self.window.show()
-
-    def openwindow(self, widget, name):
-        self.windows[name].show()
-
-    def main(self):
-        gtk.main()
-
-if __name__ == "__main__":
-    print "FreeSlot UI starting..."
-    print "Note: this will be part of paepke development, no function at the moment :)"
-    app = SlotUi()
-    app.main()
\ No newline at end of file
diff -r 313aad7435a8 -r bb150048d329 slotUI/profiles/default
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/slotUI/profiles/default	Sun Dec 02 19:19:23 2012 +0100
@@ -0,0 +1,9 @@
+[Settings]
+Name=unnamed
+Fuel=100
+Accel=10
+Limit=10
+Brake=15
+
+[Stats]
+Disabled=1
diff -r 313aad7435a8 -r bb150048d329 slotUI/slotCli.py
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/slotUI/slotCli.py	Sun Dec 02 19:19:23 2012 +0100
@@ -0,0 +1,696 @@
+#!/usr/bin/env python
+"""
+Freeslot project
+Command line interface
+"""
+
+from freeslot import Blackbox, LOGLEVEL
+from optparse import OptionParser
+from operator import itemgetter
+from subprocess import Popen, PIPE
+import sys, os
+from copy import copy
+import curses
+from time import sleep
+import ConfigParser
+
+import SimpleXMLRPCServer
+import xmlrpclib
+import threading
+
+
+VERSION = "1.8"
+MAXSLOTS = 6
+TERM = {
+    "caption": "\033[1;37m\033[1;44m",
+    "text": "\033[1;30m",
+    }
+
+# disable debug log output
+LOGLEVEL = 10
+
+SOUNDPREFIX = "quake-"
+EVENTPREFIX = "event/"
+SOUNDS = {
+        "countdown_start": os.path.abspath(SOUNDPREFIX + "sound/countdown.mp3"),
+        "race_start":      os.path.abspath(SOUNDPREFIX + "sound/racestart.mp3"),
+        "race_prepare":    os.path.abspath(SOUNDPREFIX + "sound/prepare.mp3"),
+        "lap_record":      os.path.abspath(SOUNDPREFIX + "sound/laprecord.mp3"),
+        "fuel_warning1":   os.path.abspath(SOUNDPREFIX + "sound/fuel1.mp3"),
+        "fuel_warning2":   os.path.abspath(SOUNDPREFIX + "sound/fuel2.mp3"),
+        "fuel_full":       os.path.abspath(SOUNDPREFIX + "sound/fuel_full.mp3"),
+        "pitlane_enter":   os.path.abspath(SOUNDPREFIX + "sound/pitlane_enter.mp3"),
+        "pitlane_exit":    os.path.abspath(SOUNDPREFIX + "sound/pitlane_exit.mp3"),
+        "data_error":      os.path.abspath(SOUNDPREFIX + "sound/data_error.mp3"),
+        "panic":           os.path.abspath(SOUNDPREFIX + "sound/panic.mp3"),
+        "panic_shortcut":  os.path.abspath(SOUNDPREFIX + "sound/panic_shortcut.mp3"),
+        "resume":          os.path.abspath(SOUNDPREFIX + "sound/resume.mp3"),
+        "win":             os.path.abspath(SOUNDPREFIX + "sound/win.mp3"),
+
+    }
+
+def trigger_sound(what):
+    if what in SOUNDS:
+        Popen(["/usr/bin/mpg123", "-q", SOUNDS[what]])
+        #os.spawnlp(os.P_NOWAIT, "/usr/bin/mpg123", "mpg123", SOUNDS[what])
+        #Popen(["/usr/bin/mpg123", SOUNDS[what]]).pid
+
+def trigger_event(what, slot = 0):
+    trigger_sound(what)
+    Popen(["/bin/sh", os.path.abspath(EVENTPREFIX + what), str(slot)])
+
+class SlotServer(threading.Thread):
+    def __init__(self, blackbox):
+        threading.Thread.__init__(self)
+        self.server = SimpleXMLRPCServer.SimpleXMLRPCServer(("localhost", 8000))
+        self.server.register_instance(blackbox)
+        #self.server.register_function(lambda astr: '_' + astr, '_string')
+        self.daemon = True
+
+    def run(self):
+        self.server.serve_forever()
+
+class SlotClient():
+    def __init__(self, url):
+        self.box = xmlrpclib.Server(url)
+
+class SlotCli():
+    def __init__(self, test = None, dev=""):
+        self.box = Blackbox()
+        if (not test):
+            self.box.connect(dev)
+            self.rpcserver = SlotServer(self.box)
+            self.rpcserver.start()
+        self.slot_dummy = {
+            "name": "uninitialized",
+            "laps": 0,
+            "laps_last": 0,
+            "last": 0.00,
+            "best": 0.00,
+            "fuel": 0,
+            "fuel_last": 0,
+            "position": 0,
+            "drive": 0,
+            "status": "Idle",
+            "clk": 0,
+            "car": 0,
+            "limit": 15,
+            "profilename": "default",
+            "profile": ConfigParser.RawConfigParser(),
+            }
+        self.slot_dummy["profile"].read("profiles/" + self.slot_dummy["profilename"])
+        self.slot_dummy["name"] = self.slot_dummy["profile"].get("Settings", "Name")
+        self.slot_dummy["limit"] = self.slot_dummy["profile"].getint("Settings", "Limit")
+
+        self.slot = [
+            copy(self.slot_dummy), copy(self.slot_dummy),
+            copy(self.slot_dummy), copy(self.slot_dummy),
+            copy(self.slot_dummy), copy(self.slot_dummy),
+            ]
+        self.reset_slots()
+        self.sysclk = 0.00
+        self.sysclk_last = 0.00
+        self.bestlap = 9999999.00
+        self.test = test
+        self.laplimit = 999;
+        self.timelimit = 0;
+
+    def reset_slots(self):
+        idx = 0
+        for slt in self.slot:
+            slt["laps"] = 0
+            slt["laps_last"] = 0
+            slt["last"] = 0.00
+            slt["best"] = 0.00
+            slt["fuel"] = 100
+            slt["fuel_last"] = 0
+            slt["position"] = idx
+            slt["car"] = idx # used for sort order calculation
+            slt["status"] = self.slot_dummy["status"]
+            slt["clk"] = 0
+            slt["limit"] = slt["profile"].getint("Settings", "Limit")
+            #slt["limit"] = 15
+            idx += 1
+        self.bestlap = 99999.00
+        self.raceactive = False
+
+    def update_positions(self):
+        order1 = sorted(self.slot, key=itemgetter(
+            "clk"))
+        order2 = sorted(self.slot, key=itemgetter(
+            "laps"), reverse=True)
+        idx = 1
+        for tst in order2:
+            self.slot[tst["car"]]["position"] = idx
+            idx += 1
+
+    def render_slots(self):
+        self.update_positions()
+        self.scr.addstr(3,0,
+            #"Pos | #/Name            | Laps | Best     | Last     | Fuel | Status    ",
+            "Pos | #/Name                                                            ",
+            curses.color_pair(2))
+        self.scr.addstr(4,4,
+            "  Laps | Best     | Last     | Fuel | Status                        ",
+            curses.color_pair(2))
+        for idx in range(MAXSLOTS):
+            """
+            self.scr.addstr((3 + (self.slot[idx]["position"] * 2)), 0,
+                "%3i | %i %15s | %4i | %7.2fs | %7.2fs | %3i%% | %10s" % (
+                self.slot[idx]["position"],
+                self.slot[idx]["car"] + 1, self.slot[idx]["name"],
+                self.slot[idx]["laps"],
+                self.slot[idx]["best"],
+                self.slot[idx]["last"],
+                self.slot[idx]["fuel"],
+                self.slot[idx]["status"],
+                ),
+                curses.color_pair(11 + idx) )
+            """
+            if idx > 3:
+                namesuffix = " (%i)" % self.slot[idx]["drive"]
+            else:
+                namesuffix = ""
+
+            self.scr.addstr((3 + (self.slot[idx]["position"] * 2)), 0,
+                "%3i | %i %15s %48s" % (
+                self.slot[idx]["position"],
+                self.slot[idx]["car"] + 1, (self.slot[idx]["name"] + namesuffix),
+                "",
+                ),
+                curses.color_pair(11 + idx) )
+            self.scr.addstr((4 + (self.slot[idx]["position"] * 2)), 4,
+                "  %4i | %7.2fs | %7.2fs | %3i%% | %10s | %2i% 15s" % (
+                self.slot[idx]["laps"],
+                self.slot[idx]["best"],
+                self.slot[idx]["last"],
+                self.slot[idx]["fuel"],
+                self.slot[idx]["status"],
+                self.slot[idx]["limit"],
+                ""
+                ),
+                curses.color_pair(11 + idx) )
+
+    def cleartop(self):
+        self.scr.addstr(0,0, "%60s" % "Race Limits: %i Laps / %i Minutes" % (self.laplimit, self.timelimit))
+        self.scr.addstr(1,0, "%80s" % " ")
+
+    def flash_car_settings(self, slot):
+        # write current settings to car firmware
+        self.box.setmode(0)
+        self.cleartop()
+        self.scr.addstr(0,0, "Writing settings for %s to car %i..." % (
+            self.slot[slot]["name"],
+            slot + 1),
+            curses.color_pair(1))
+        self.scr.refresh()
+        self.box.progcar(slot, "fuel", 0)
+        sleep(0.5)
+        self.box.progcar(slot, "accel", self.slot[slot]["profile"].getint("Settings", "Accel"))
+        sleep(0.5)
+        self.box.progcar(slot, "brake", self.slot[slot]["profile"].getint("Settings", "Brake"))
+        sleep(0.5)
+        self.box.speedlimit(slot, self.slot[slot]["limit"])
+        sleep(0.5)
+        self.cleartop()
+        self.box.setmode(1)
+
+    def readName(self, slot):
+        self.scr.nodelay(0) # enable delay on readkey
+        curses.echo()
+        self.scr.addstr(0,0, "Enter Name for Controller %i [%s]:" % (
+            slot + 1,
+            self.slot[slot]["name"]),
+            curses.color_pair(1))
+        self.scr.refresh()
+        name = self.scr.getstr(1,0, 15)
+        if name != "":
+            # look if profile with that name found
+            try:
+                with open("profiles/" + name) as f: pass
+                self.slot[slot]["profilename"] = name
+                self.slot[slot]["profile"].read("profiles/" + name)
+                self.slot[slot]["name"] = self.slot[slot]["profile"].get("Settings", "Name")
+                self.slot[slot]["limit"] = self.slot[slot]["profile"].getint("Settings", "Limit")
+                self.flash_car_settings(slot)
+            except IOError, err:
+                self.slot[slot]["profilename"] = "default"
+                self.slot[slot]["profile"].read("profiles/default")
+                self.slot[slot]["limit"] = self.slot[slot]["profile"].getint("Settings", "Limit")
+                self.slot[slot]["name"] = name
+                self.flash_car_settings(slot)
+        self.cleartop()
+        self.scr.refresh()
+        curses.noecho()
+        self.scr.nodelay(1) # disable delay on readkey
+
+    def readLimit(self, slot):
+        limit = self.readInt("SPEEDLIMIT for %s (%i)" % (
+            self.slot[slot]["name"],
+            slot + 1),
+            self.slot[slot]["limit"], 15)
+        if limit:
+            self.slot[slot]["limit"] = limit
+            self.cleartop()
+            self.box.speedlimit(slot, limit)
+
+
+    def readInt(self, msg, default, maximum = 999999):
+        self.scr.nodelay(0) # enable delay on readkey
+        curses.echo()
+        self.scr.addstr(0,0, "%s [%i]:" % (
+            msg,
+            default),
+            curses.color_pair(1))
+        self.scr.refresh()
+        inp = self.scr.getstr(1,0, 4)
+        if inp != "":
+            try:
+                inp = int(inp)
+                if inp > maximum: inp = maximum
+            except Exception:
+                inp = None
+        else:
+            inp = None
+        self.cleartop()
+        self.scr.refresh()
+        curses.noecho()
+        self.scr.nodelay(1) # disable delay on readkey
+        return inp
+
+
+    def monitor_init(self, live = 1):
+        """
+        Send initializing commands for live monitoring
+        """
+        self.box.query("F1\n") # set fuel logic enabled
+        self.box.query("*%i\n" % live) # set live fuel info
+
+    def monitor_learn(self, slot):
+        # clear garbage in UART rx buffer
+        self.box.query("*0\n") # set live fuel info
+        self.box.query("*0\n") # set live fuel info
+        while self.box.readline() != "": pass
+
+        trk = False
+        spd = 0
+        trk_old = False
+        spd_old = 0
+        clock = -1
+
+        self.monitor_init(slot + 2)
+        while 1:
+            #key = self.scr.getch()
+            #if key == ord('c'): break
+
+            # is there something in the rx buffer?
+            rx = self.box.readline()
+            if (rx != ""):
+                try:
+                    data = rx.split(":")
+                    if rx[:3] == "LN:":
+                        if clock >= 0:
+                            clock += 1
+                        spd = int(data[1], 16)
+                        trk = (data[2] != 'X')
+                        if (spd != spd_old) or (trk != trk_old):
+                            if clock < 0:
+                                clock = 0
+                            print "%i,%i,%s" % (clock, spd, trk)
+                        trk_old = trk
+                        spd_old = spd * 1
+                    if rx[:2] == "L:":
+                        # update lap time info
+                        l = int(data[2], 16)
+                        s = int(data[3]) - 1
+                        t = int(data[4], 16) / 2000.00
+                        if (slot == s):
+                            print "# lap %i complete: %3.2f seconds" % (l, t)
+                            clock = 0
+                            print "%i,%i,%s" % (clock, spd, trk)
+                except:
+                    print "RX ERROR: " % rx
+
+    def monitor_playback(self, slot, filename):
+        # clear garbage in UART rx buffer
+        self.box.query("*0\n") # set live fuel info
+        self.box.query("*0\n") # set live fuel info
+        sleep(1)
+        cli.box.speedminimum(slot, 0 )
+        while self.box.readline() != "": pass
+
+        clock = -5
+        trkfile = open(filename, "r").readlines()
+        print "Loading %s..." % filename
+
+        while 1:
+            try:
+                for l in trkfile:
+                    l = l.strip()
+                    if (l != "") and (l[:1] != "#"):
+                        print "Line: %s" % repr(l)
+                        data = l.split(",")
+                        speed = int(data[1])
+                        while (clock < int(data[0]) and (int(data[0]) > 0)):
+                            clock += 1
+                            sleep(0.07)
+                        print "CLK %i/%i -> set: %i" % (clock, int(data[0]), speed)
+                        cli.box.speedminimum(slot, speed )
+                # now wait for lap sync :)
+                while self.box.readline() != "": pass
+                rx = ""
+                while rx[:2] != "L:":
+                    rx = self.box.readline()
+                data = rx.split(":")
+                l = int(data[2], 16)
+                s = int(data[3]) - 1
+                t = int(data[4], 16) / 2000.00
+                print "# lap %i complete: %3.2f seconds" % (l, t)
+                clock = -3
+            except Exception, e:
+                print repr(e)
+                sys.exit(1)
+            except KeyboardInterrupt:
+                print "resetting"
+                cli.box.speedminimum(slot, 0 )
+                cli.box.speedminimum(slot, 0 )
+                sys.exit(0)
+                
+
+
+    def monitor(self):
+        """
+        Live Monitor on the console
+        Keyboard loop to control it???
+        """
+        # clear garbage in UART rx buffer
+        while self.box.readline() != "": pass
+
+        self.monitor_init()
+        self.scr = curses.initscr()
+        curses.start_color()
+        curses.init_pair(1, curses.COLOR_WHITE, curses.COLOR_BLACK) # standard text
+        curses.init_pair(2, curses.COLOR_WHITE, curses.COLOR_BLUE) # label
+        curses.init_pair(11, curses.COLOR_BLACK, curses.COLOR_YELLOW) # player 1 slot
+        curses.init_pair(12, curses.COLOR_BLACK, curses.COLOR_GREEN) # player 2 slot
+        curses.init_pair(13, curses.COLOR_BLACK, curses.COLOR_RED) # player 3 slot
+        curses.init_pair(14, curses.COLOR_BLACK, curses.COLOR_MAGENTA) # player 4 slot
+        curses.init_pair(15, curses.COLOR_WHITE, curses.COLOR_BLACK) # player 5 slot
+        curses.init_pair(16, curses.COLOR_WHITE, curses.COLOR_BLACK) # player 6 slot
+        curses.noecho() # disable key echo
+        curses.cbreak() # do not buffer keypresses
+        self.scr.keypad(1) # enable special keys
+        self.scr.nodelay(1) # disable delay on readkey
+
+        self.cleartop()
+        self.render_slots()
+        self.scr.refresh()
+
+
+        while 1:
+            key = self.scr.getch()
+            if key == ord('c'): break
+            elif key == ord(' '): self.box.query("+") # panic / resume
+            elif key == 10: self.box.query("#") # remote start button press
+            elif key == ord('1'): self.readName(0)
+            elif key == ord('2'): self.readName(1)
+            elif key == ord('3'): self.readName(2)
+            elif key == ord('4'): self.readName(3)
+            elif key == ord('5'): self.readName(4)
+            elif key == ord('6'): self.readName(5)
+            #elif key == ord('q'): self.readLimit(0)
+            #elif key == ord('w'): self.readLimit(1)
+            #elif key == ord('e'): self.readLimit(2)
+            #elif key == ord('r'): self.readLimit(3)
+            #elif key == ord('t'): self.readLimit(4)
+            #elif key == ord('z'): self.readLimit(5)
+            elif key == ord('a'):
+                if self.slot[4]["drive"] > 0: self.slot[4]["drive"] -= 1
+                cli.box.speedminimum(4, self.slot[4]["drive"])
+            elif key == ord('s'):
+                if self.slot[4]["drive"] < 16: self.slot[4]["drive"] += 1
+                cli.box.speedminimum(4, self.slot[4]["drive"])
+            elif key == ord('y'):
+                if self.slot[5]["drive"] > 0: self.slot[5]["drive"] -= 1
+                cli.box.speedminimum(5, self.slot[5]["drive"])
+            elif key == ord('x'):
+                if self.slot[5]["drive"] < 16: self.slot[5]["drive"] += 1
+                cli.box.speedminimum(5, self.slot[5]["drive"])
+            elif key == ord('t'):
+                tmp = self.readInt("Set new Race TIME limit", self.timelimit)
+                if tmp: self.timelimit = tmp
+                self.cleartop()
+            elif key == ord('l'):
+                tmp = self.readInt("Set new Race LAP limit", self.laplimit)
+                if tmp: self.laplimit = tmp
+                self.cleartop()
+            elif key == ord('/'):
+                for slot in range(5):
+                    self.flash_car_settings(slot)
+
+
+            # is there something in the rx buffer?
+            rx = self.box.readline()
+            if (rx != "") or self.test:
+                self.scr.addstr(17,0,
+                    "Last RX: %19s" % rx, curses.color_pair(2))
+                self.scr.redrawwin()
+                self.scr.refresh()
+                # we have received something
+                try:
+                    data = rx.split(":")
+                    if rx[:2] == "L:":
+                        # update lap time info
+                        l = int(data[2], 16)
+                        slot = int(data[3]) - 1
+                        t = int(data[4], 16) / 2000.00
+                        self.sysclk = int(data[5], 16) / 2000.00
+                        self.slot[slot]["laps_last"] = self.slot[slot]["laps"]
+                        self.slot[slot]["laps"] = l
+                        self.slot[slot]["last"] = t
+                        self.slot[slot]["clk"] = self.sysclk
+                        if (self.slot[slot]["best"] > t) or (self.slot[slot]["best"] == 0):
+                            self.slot[slot]["best"] = t
+                        if self.bestlap > t:
+                            trigger_event("lap_record", slot + 1)
+                            self.bestlap = t
+
+                        self.slot[slot]["status"] = "IN-RACE"
+                        if (self.slot[slot]["laps_last"] != l) and (l == self.laplimit):
+                            # we have lap limit reached!
+                            trigger_event("win", slot + 1)
+                            self.raceactive = False
+                            self.slot[slot]["status"] = "WINNER!"
+                            self.box.query("+") # stop race
+
+                        self.render_slots()
+
+                    if rx[:2] == "F:":
+                        # update fuel level
+                        slot = int(data[1])
+                        f = int(data[2], 16)
+                        f = f / 100 # fuel in percent
+                        self.sysclk = int(data[3], 16) / 2000.00
+                        self.slot[slot]["fuel_last"] = self.slot[slot]["fuel"]
+                        self.slot[slot]["fuel"] = f
+                        if self.slot[slot]["fuel_last"] != f:
+                            if (self.slot[slot]["fuel_last"] == 16) and (f == 15):
+                                # 15 percent fuel, set speed limit for car to 8
+                                # warning sound
+                                trigger_event("fuel_warning1", slot + 1)
+                                cli.box.speedlimit(slot, 8)
+                            if (self.slot[slot]["fuel_last"] == 6) and (f == 5):
+                                # 5 percent, set speed limit for car to 6
+                                # warning sound
+                                trigger_event("fuel_warning2", slot + 1)
+                                cli.box.speedlimit(slot, 6)
+                            if (self.slot[slot]["fuel_last"] == 1) and (f == 0):
+                                # fuel empty
+                                # set speedlimit to 4
+                                cli.box.speedlimit(slot, 4)
+                            if (self.slot[slot]["fuel_last"] < f) and (f >= 11) and (f < 20):
+                                cli.box.speedlimit(slot, 15)
+                            if (self.slot[slot]["fuel_last"] < f) and (f == 100):
+                                trigger_event("fuel_full", slot + 1)
+                        self.render_slots()
+
+                    if rx[:1] == "~":
+                        # jumpstart occured
+                        slot = int(rx[1:2])
+                        t = int(data[1], 16) / 2000.00
+                        self.slot[slot]["jumpstart"] = t
+                        self.slot[slot]["status"] = "Jumpstart!"
+
+                    if rx[:3] == "RW:":
+                        # ResponseWire packet, do nothing at the moment, just decode
+                        slot = int(data[1])
+                        devtype = int(data[2])
+                        sender = int(data[3], 16)
+                        status = int(data[4], 16)
+                        self.sysclk = int(data[5], 16)
+                        if (devtype == 4):
+                            # pitlane sent something
+                            if (status == 5):
+                                self.slot[slot]["status"] = "PITLANE"
+                                trigger_event("pitlane_enter", slot + 1)
+                            if (status == 7):
+                                self.slot[slot]["status"] = "IN-RACE"
+                                trigger_event("pitlane_exit", slot + 1)
+
+                        self.render_slots()
+
+                    if rx == "!RACE PREPARE":
+                        # reset current race status
+                        # and display PREPARE PHASE
+                        self.reset_slots()
+                        for slot in range(MAXSLOTS):
+                            self.slot[slot]["status"] = "Prepare"
+                        trigger_event("race_prepare")
+
+                    if rx == "!RACE START":
+                        for slot in range(MAXSLOTS):
+                            if self.slot[slot]["status"] == "~~~~~~~~~~":
+                                self.slot[slot]["status"] = "Idle"
+                        trigger_event("race_start")
+                        self.raceactive = True
+
+                    if rx == "!COUNTDOWN":
+                        # countdown initiated
+                        for slot in range(MAXSLOTS):
+                            self.slot[slot]["status"] = "~~~~~~~~~~"
+                        trigger_event("countdown_start")
+
+                    if rx == "!PANIC":
+                        # panic mode
+                        trigger_event("panic")
+
+                    if rx == "!SHORTCUT":
+                        # panic mode
+                        trigger_event("panic_shortcut")
+
+                    if rx == "!RESUME":
+                        # panic mode
+                        trigger_event("resume")
+
+
+                    if ((self.timelimit > 0) and (self.raceactive) and 
+                        (self.sysclk_last != self.sysclk) and 
+                        ((self.sysclk / 60) >= self.timelimit)):
+                        self.sysclk_last = self.sysclk
+                        self.raceactive = False
+                        # we have time limit reached!
+                        self.box.query("+") # stop race
+                        trigger_event("win")
+                        # get the one with position 1
+                        for slot in self.slots:
+                            if slot["position"] == 1:
+                                slot["status"] = "WINNER!"
+                	self.render_slots()
+                    self.sysclk_last = self.sysclk
+    
+
+
+                    self.scr.addstr(17,31,
+                        "Race Timer: %7.3f min" % (self.sysclk / 60),
+                        curses.color_pair(2))
+
+                    self.scr.refresh()
+
+                except Exception:
+                    trigger_event("data_error")
+                    pass
+
+        # terminate
+        curses.nocbreak()
+        self.scr.keypad(0)
+        curses.echo()
+        curses.endwin()
+        return None
+
+    def cyclemode(self):
+        pass
+
+if __name__ == "__main__":
+    parser = OptionParser(version="%prog " + VERSION)
+    parser.add_option("--live", dest="live", action="store_true", default=False,
+        help="Run Live monitor on console", metavar="[0-5]")
+    parser.add_option("--learn", dest="learn", action="store_true", default=False,
+        help="Run Learning mode for [slot]", metavar="[0-5]")
+    parser.add_option("--teach", dest="playback", 
+        help="Playback teach file", metavar="[filename]")
+
+    parser.add_option("--slot", dest="carid",
+        help="Required for programming a car directly", metavar="[1-6]")
+    parser.add_option("--fuel", dest="fuel",
+        help="Set maximum CAR fuel level", metavar="[0-15]")
+    parser.add_option("--brake", dest="brake",
+        help="Set CAR brake strength", metavar="[0-15]")
+    parser.add_option("--accel", dest="accel",
+        help="Set CAR acceleration ", metavar="[6-15]")
+    parser.add_option("--blink", dest="blink",
+        help="Set car lights blinking state", metavar="[on|off]")
+    parser.add_option("--limit", dest="limit",
+        help="Controlled SPEED LIMIT (15 = no limit)", metavar="[0-15]")
+    parser.add_option("--drive", dest="drive",
+        help="Controlled SPEED MINIMUM (0 = disabled)", metavar="[0-15]")
+    parser.add_option("--test", dest="test", action="store_true", default=False,
+        help="", metavar="")
+    parser.add_option("--dev", dest="dev", default="/dev/ttyUSB0",
+        help="Communication port", metavar="[/dev/ttyUSB0]")
+
+    (options, args) = parser.parse_args()
+    #if not options.dev:
+    #    options.dev = "/dev/ttyUSB0"
+
+    if options.live or options.learn or options.playback:
+        cli = SlotCli(options.test, options.dev)
+    else:
+        cli = SlotClient('http://localhost:8000')
+    # should a CLI function be started?
+
+    if options.live:
+        # start the live monitor
+        cli.monitor()
+        sys.exit(0)
+
+    # check commandline if we have to program something
+    if not options.carid:
+        print "Option --slot is required for all car programming commands!\nUse --help to get a list of available commands"
+        sys.exit(1)
+    else:
+        options.carid = int(options.carid) - 1
+        if (options.carid < 0) or (options.carid > 6):
+            print "Error: Invalid slot selected"
+            sys.exit(1)
+
+    if options.learn:
+        # start the learn monitor
+        cli.monitor_learn(options.carid)
+        sys.exit(0)
+
+    if options.playback:
+        # start the playback monitor
+        cli.monitor_playback(options.carid, options.playback)
+        sys.exit(0)
+
+    if options.fuel:
+        print "setFuel: " + cli.box.progcar(int(options.carid), "fuel", int(options.fuel))
+
+    if options.accel:
+        print "setAccel: " + cli.box.progcar(int(options.carid), "accel", int(options.accel))
+
+    if options.brake:
+        print "setBrake: " + cli.box.progcar(int(options.carid), "brake", int(options.brake))
+
+    if options.blink:
+        state = False
+        if options.blink == "on":
+            state = True
+        print "setBlink: " + cli.box.blinkcar(int(options.carid), state)
+
+    if options.limit:
+        print "Change Speed Limit: " + cli.box.speedlimit(int(options.carid), int(options.limit))
+
+    if options.drive:
+        print "Change minimum Speed drive: " + cli.box.speedminimum(int(options.carid), int(options.drive))
+