fix: don't close client sockets too early
[saartuer.git] / tyshell
diff --git a/tyshell b/tyshell
index f30b9d797928f06d10dda8c3638480c04df70d11..70ad6df0746316232e8f15a72dead40663215573 100755 (executable)
--- a/tyshell
+++ b/tyshell
@@ -7,6 +7,8 @@ import subprocess
 import socket
 import pwd
 import grp
 import socket
 import pwd
 import grp
+import traceback
+from collections import namedtuple
 
 tuerSock = "/run/tuer.sock"
 
 
 tuerSock = "/run/tuer.sock"
 
@@ -22,7 +24,7 @@ atexit.register(readline.write_history_file, histfile)
 # available commands
 def helpcmd(c):
        if (len(c) > 1):
 # available commands
 def helpcmd(c):
        if (len(c) > 1):
-               print(commands.get(c[1],(None,'Can\'t find help for command %s'%(c[1])))[1])
+               print(commands.get(c[1],(None,'Can\'t find help for command %s'%(c[1]))).helpstring)
        else:
                print("Available commands: %s" % ", ".join(sorted(commands.keys())))
                print("Use 'help command' to get more information on the command 'command'")
        else:
                print("Available commands: %s" % ", ".join(sorted(commands.keys())))
                print("Use 'help command' to get more information on the command 'command'")
@@ -41,9 +43,11 @@ def sendcmd(addr, cmd):
                s.connect(addr)
                s.settimeout(60.0)
                s.send(cmd.encode())
                s.connect(addr)
                s.settimeout(60.0)
                s.send(cmd.encode())
-               data = s.recv(256)
+               while True:
+                       data = s.recv(256)
+                       if not len(data): break
+                       print(data.decode('utf-8'))
                s.close()
                s.close()
-               print(data.decode('utf-8'))
        return run
 
 def exitcmd(c):
        return run
 
 def exitcmd(c):
@@ -60,15 +64,31 @@ def alias (cmds, aliases):
                cmds[newname] = cmds[oldname]
        return cmds
 
                cmds[newname] = cmds[oldname]
        return cmds
 
+def prompt_sure(f,msg):
+       def run(c):
+               try:
+                       command = input("%s Are you sure? (yes/no) > " % msg)
+               except EOFError:
+                       print()
+                       return
+               if command[0] == 'y':
+                       return f(c)
+       return run
+
+CmdEntry = namedtuple('CmdEntry','function helpstring')
+
 commands = alias({
 commands = alias({
-       'exit': (exitcmd, 'Quits this shell'),
-       'help': (helpcmd, 'Helps you getting to know the available commands'),
-       'open': (sendcmd(tuerSock, 'unlock'), 'Will try to unlock the apartment door'),
-       'buzz': (sendcmd(tuerSock, 'buzz'), 'Will buzz the buzzer for the street door'),
-       'who': (whocmd, 'Shows the list of people, who are allowed to control this system'),
+       'exit': CmdEntry(exitcmd, 'Quits this shell'),
+       'help': CmdEntry(helpcmd, 'Helps you getting to know the available commands'),
+       'unlock': CmdEntry(sendcmd(tuerSock, 'unlock'), 'Will try to unlock the apartment door'),
+       'lock': CmdEntry(sendcmd(tuerSock, 'lock'), 'If in fallback mode, try to lock the apartment door. If not in fallback mode, you must use the switch near the door.'),
+       'buzz': CmdEntry(sendcmd(tuerSock, 'buzz'), 'Will buzz the buzzer for the street door'),
+       'who': CmdEntry(whocmd, 'Shows the list of people, who are allowed to control this system'),
+       'fallback_mode_on': CmdEntry(prompt_sure(sendcmd(tuerSock, 'fallback_mode_on'),'WARNING: This action will be reported to the admins. Use this only in case of Sphinx hardware failure when you need to ignore erroneous sensor input!'), 'Sets the system in a state where it is less dependent on sensoric input. Use it only when sensors are broken.'),
+       'fallback_mode_off': CmdEntry(prompt_sure(sendcmd(tuerSock, 'fallback_mode_off'),'WARNING: This action will be reported to the admins. Use this only if you have fixed the sensors of the Sphinx or activated fallback mode by accident!'), 'Resets the system to the default state. Use this when you have just repaired the sensors of the Sphinx.'),
 },{
        # aliases
 },{
        # aliases
-       'unlock': 'open',
+       'open': 'unlock',
 })
 
 def complete_command(cmd):
 })
 
 def complete_command(cmd):
@@ -97,10 +117,11 @@ while True:
                print("Command %s not found. Use help." % command[0])
        elif len(cmdoptions) == 1: # exactly one command fits (prefix)
                try:
                print("Command %s not found. Use help." % command[0])
        elif len(cmdoptions) == 1: # exactly one command fits (prefix)
                try:
-                       res = commands[cmdoptions[0]][0](command)
+                       res = commands[cmdoptions[0]].function(command)
                        if res: break
                except Exception as e:
                        print("Error while executing %s: %s" % (command[0], str(e)))
                        if res: break
                except Exception as e:
                        print("Error while executing %s: %s" % (command[0], str(e)))
+                       #print(traceback.format_exc())
        else: # multiple commands fit the prefix
                print("Ambiguous command prefix, please choose one of the following:")
                print("\t", " ".join(cmdoptions))
        else: # multiple commands fit the prefix
                print("Ambiguous command prefix, please choose one of the following:")
                print("\t", " ".join(cmdoptions))