#!/usr/bin/python3 import os import readline import shlex import sys import subprocess import socket import pwd import grp from collections import namedtuple tuerSock = "/run/tuer.sock" # use a histfile histfile = os.path.join(os.path.expanduser("~"), ".tyshellhist") try: readline.read_history_file(histfile) except IOError: pass import atexit atexit.register(readline.write_history_file, histfile) # available commands def helpcmd(c): if (len(c) > 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'") def extcmd(cmd): def run(c): ret = subprocess.call(cmd) if ret != 0: print("Command returned non-zero exit statis %d" % ret) return run def sendcmd(addr, cmd): def run(c): print("206 Sending command %s..." % (cmd)) s = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM) s.connect(addr) s.settimeout(60.0) s.send(cmd.encode()) data = s.recv(256) s.close() print(data.decode('utf-8')) return run def exitcmd(c): print("Bye") return True def whocmd(c): for n in grp.getgrnam("tuer").gr_mem: p = pwd.getpwnam(n) print (p.pw_name, " - ", p.pw_gecos) def alias (cmds, aliases): for newname, oldname in aliases.items(): cmds[newname] = cmds[oldname] return cmds CmdEntry = namedtuple('CmdEntry','function helpstring') commands = alias({ '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'), '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'), },{ # aliases 'open': 'unlock', }) def complete_command(cmd): '''returns a list of commands (as strings) starting with cmd''' return list(filter(lambda x: x.startswith(cmd), commands.keys())) readline.set_completer(lambda cmd, num: (complete_command(cmd)+[None])[num]) # wrap complete_command for readline's weird completer API readline.parse_and_bind("tab: complete") # run completion on tab # input loop print("Welcome to tyshell. Use help to see what you can do.") while True: try: command = input("$ ") except EOFError: print() break command = shlex.split(command) if not len(command): continue # find suiting commands if command[0] in commands: # needed in case a complete command is a prefix of another one cmdoptions = [command[0]] else: cmdoptions = complete_command(command[0]) # check how many we found if len(cmdoptions) == 0: # no commands fit prefix print("Command %s not found. Use help." % command[0]) elif len(cmdoptions) == 1: # exactly one command fits (prefix) try: res = commands[cmdoptions[0]].function(command) if res: break except Exception as e: print("Error while executing %s: %s" % (command[0], str(e))) else: # multiple commands fit the prefix print("Ambiguous command prefix, please choose one of the following:") print("\t", " ".join(cmdoptions)) # TODO: put current "command[0]" into the shell for the next command, but such that it is deletable with backspace