Browse Source

Refactored RCON shell.

pull/2/head 1.1.8
Richard Neumann 4 years ago
parent
commit
af6ef6aeec
  1. 183
      rcon/console.py
  2. 5
      rcon/rconshell.py

183
rcon/console.py

@ -1,7 +1,6 @@
"""An interactive console.""" """An interactive console."""
from getpass import getpass from getpass import getpass
from typing import Union
from rcon.config import Config from rcon.config import Config
from rcon.exceptions import RequestIdMismatch, WrongPassword from rcon.exceptions import RequestIdMismatch, WrongPassword
@ -12,131 +11,133 @@ __all__ = ['rconcmd']
EXIT_COMMANDS = {'exit', 'quit'} EXIT_COMMANDS = {'exit', 'quit'}
MSG_QUERY_LATER = '\nOkay, I will ask again later.'
MSG_ABORTED = '\nAborted...'
MSG_LOGIN_ABORTED = '\nLogin aborted. Bye.' MSG_LOGIN_ABORTED = '\nLogin aborted. Bye.'
MSG_EXIT = 'Bye.' MSG_EXIT = '\nBye.'
MSG_SESSION_TIMEOUT = 'Session timed out. Please login again.' MSG_SESSION_TIMEOUT = 'Session timed out. Please login again.'
MSG_EXIT_USAGE = 'Usage: {} [<exit_code>].'
PROMPT = 'RCON> ' PROMPT = 'RCON> '
def read(prompt: str, typ: type = None) -> type: def read_host() -> str:
"""Reads input and converts it to the respective type.""" """Reads the host."""
while True: while True:
raw = input(prompt) try:
return input('Host: ')
except KeyboardInterrupt:
continue
if typ is not None:
try:
return typ(raw)
except (TypeError, ValueError):
print(f'Invalid {typ}: {raw}')
continue
return raw def read_port() -> int:
"""Reads the port."""
while True:
try:
port = input('Port: ')
except KeyboardInterrupt:
continue
def read_or_none(prompt: str, typ: type = None) -> type: try:
"""Reads the input and returns None on EOFError.""" port = int(port)
except KeyboardInterrupt:
print(f'Invalid integer: {port}')
continue
try: if 0 <= port <= 65535:
return read(prompt, typ=typ) return port
except EOFError:
print(MSG_QUERY_LATER)
return None
print(f'Invalid port: {port}')
continue
def login(client: Client, passwd: str) -> str:
"""Performs a login.""" def read_passwd() -> str:
"""Reads the password."""
while True:
try:
return getpass('Password: ')
except KeyboardInterrupt:
continue
def get_config(host: str, port: int, passwd: str) -> Config:
"""Reads the necessary arguments."""
if host is None:
host = read_host()
if port is None:
port = read_port()
if passwd is None: if passwd is None:
passwd = getpass('Password: ') passwd = read_passwd()
return Config(host, port, passwd)
logged_in = False
while not logged_in: def login(client: Client, passwd: str) -> str:
"""Performs a login."""
while True:
try: try:
logged_in = client.login(passwd) client.login(passwd)
except WrongPassword: except WrongPassword:
print('Invalid password.') print('Invalid password.')
passwd = getpass('Password: ') passwd = read_passwd()
continue
return passwd return passwd
def get_config(host: str, port: int, passwd: str) -> Config: def process_input(client: Client, passwd: str, prompt: str) -> bool:
"""Reads the necessary arguments.""" """Processes the CLI input."""
while any(item is None for item in (host, port, passwd)):
if host is None:
host = read_or_none('Host: ')
if port is None: try:
port = read_or_none('Port: ', typ=int) command = input(prompt)
except KeyboardInterrupt:
print()
return True
except EOFError:
print(MSG_EXIT)
return False
if passwd is None: try:
passwd = getpass('Password: ') command, *args = command.split()
except ValueError:
return True
return Config(host, port, passwd) if command in EXIT_COMMANDS:
return False
try:
result = client.run(command, *args)
except RequestIdMismatch:
print(MSG_SESSION_TIMEOUT)
def exit(exit_code: Union[int, str] = 0) -> int: # pylint: disable=W0622 try:
"""Exits the interactive shell via exit command.""" passwd = login(client, passwd)
except EOFError:
print(MSG_LOGIN_ABORTED)
return False
print(MSG_EXIT) print(result)
return int(exit_code) return True
def rconcmd(host: str, port: int, passwd: str, *, prompt: str = PROMPT) -> int: def rconcmd(host: str, port: int, passwd: str, *, prompt: str = PROMPT):
"""Initializes the console.""" """Initializes the console."""
try: try:
config = get_config(host, port, passwd) host, port, passwd = get_config(host, port, passwd)
except KeyboardInterrupt: except EOFError:
print(MSG_ABORTED) print(MSG_EXIT)
return 1 return
with Client(config.host, config.port) as client: with Client(host, port) as client:
try: try:
passwd = login(client, config.passwd) passwd = login(client, passwd)
except (EOFError, KeyboardInterrupt): except EOFError:
print(MSG_LOGIN_ABORTED) print(MSG_LOGIN_ABORTED)
return 1 return
while True: while process_input(client, passwd, prompt):
try: pass
command = input(prompt)
except EOFError:
print(f'\n{MSG_EXIT}')
break
except KeyboardInterrupt:
print()
continue
try:
command, *args = command.split()
except ValueError:
continue
if command in EXIT_COMMANDS:
try:
return exit(*args) # pylint: disable=R1722
except (TypeError, ValueError):
print(MSG_EXIT_USAGE.format(command))
continue
try:
result = client.run(command, *args)
except RequestIdMismatch:
print(MSG_SESSION_TIMEOUT)
try:
passwd = login(client, passwd)
except (EOFError, KeyboardInterrupt):
print(MSG_LOGIN_ABORTED)
return 2
print(result)
return 0

5
rcon/rconshell.py

@ -4,7 +4,6 @@ from argparse import ArgumentParser, Namespace
from logging import INFO, basicConfig, getLogger from logging import INFO, basicConfig, getLogger
from pathlib import Path from pathlib import Path
from socket import timeout from socket import timeout
from sys import exit # pylint: disable=W0622
from rcon.errorhandler import ErrorHandler from rcon.errorhandler import ErrorHandler
from rcon.exceptions import RequestIdMismatch from rcon.exceptions import RequestIdMismatch
@ -49,6 +48,4 @@ def main() -> None:
with ErrorHandler(ERRORS, LOGGER): with ErrorHandler(ERRORS, LOGGER):
with CommandHistory(LOGGER): with CommandHistory(LOGGER):
returncode = rconcmd(host, port, passwd, prompt=args.prompt) rconcmd(host, port, passwd, prompt=args.prompt)
exit(returncode)

Loading…
Cancel
Save