sync-function support and ask / print interface; updated TODO

This commit is contained in:
Dominik Moritz Roth 2020-07-19 14:08:54 +02:00
parent fa1861ff27
commit 8410e7afbe
2 changed files with 34 additions and 8 deletions

38
Rex.py
View File

@ -8,6 +8,7 @@ from prompt_toolkit.completion import Completion
from prompt_toolkit.auto_suggest import AutoSuggestFromHistory from prompt_toolkit.auto_suggest import AutoSuggestFromHistory
from fuzzywuzzy import fuzz from fuzzywuzzy import fuzz
from shlex import split from shlex import split
import pprint
# @@@@ # @@@@
# &@((((((((@@&(@@@ # &@((((((((@@&(@@@
@ -83,6 +84,8 @@ async def arg2(arg1, arg2):
print("got " + arg1 + " and " + arg2) print("got " + arg1 + " and " + arg2)
async def arg4(arg1, arg2, arg3, arg4): async def arg4(arg1, arg2, arg3, arg4):
print("got " + arg1 + " and " + arg2 + " and " + arg3 + " and " + arg4) print("got " + arg1 + " and " + arg2 + " and " + arg3 + " and " + arg4)
async def question():
await rex.ask("What is your question? ")
# This is an example (the default) cmd-dict # This is an example (the default) cmd-dict
@ -93,6 +96,7 @@ defaultCmds = {
"arg": arg, "arg": arg,
"arg2": arg2, "arg2": arg2,
"arg4": arg4, "arg4": arg4,
"question": question,
"nested": { "nested": {
"a": nA, "a": nA,
"b": nB, "b": nB,
@ -153,15 +157,15 @@ class _CompletionLookup(Completer):
class Rex(): class Rex():
def __init__(self, cmds=defaultCmds, prompt="[~> ", hasToolbar = True, def __init__(self, cmds=defaultCmds, prompt="[~> ", hasToolbar = True,
printExceptions = True, raiseExceptions = False, pipeReturn = False): printExceptions = True, raiseExceptions = False):
self.cmds = cmds self.cmds = cmds
self.prompt = prompt self.prompt = prompt
self.askPrompt = "[?> "
self.hasToolbar = hasToolbar self.hasToolbar = hasToolbar
self.session = PromptSession() self.session = PromptSession()
self.toolbar = [("", "")] self.toolbar = [("", "")]
self.printExceptions = printExceptions self.printExceptions = printExceptions
self.raiseExceptions = raiseExceptions self.raiseExceptions = raiseExceptions
self.pipeReturn = pipeReturn
async def once(self): async def once(self):
with patch_stdout(): with patch_stdout():
@ -187,9 +191,13 @@ class Rex():
print("[!] The given commands expects "+str(len(inspect.getfullargspec(pos)[0]))+" arguments, "+ print("[!] The given commands expects "+str(len(inspect.getfullargspec(pos)[0]))+" arguments, "+
"but "+str(len(words[index+1:]))+" were given") "but "+str(len(words[index+1:]))+" were given")
else: else:
ret = await pos(*words[index+1:]) # run the function
if self.pipeReturn: if inspect.iscoroutinefunction(pos):
return ret # function is async
ret = await pos(*words[index+1:])
else:
ret = pos(*words[index+1:])
await self.print(ret)
else: else:
print("[!] No such command") print("[!] No such command")
except Exception as e: except Exception as e:
@ -202,11 +210,27 @@ class Rex():
return True return True
async def run(self): async def run(self):
if self.pipeReturn:
raise Exception("Cannot 'run', if pipeReturn is set to true")
while await self.once(): while await self.once():
pass pass
async def print(self, txt):
if txt==None:
return
nice = pprint.pformat(txt)
for line in nice.split("\n"):
print("[<] "+line)
async def ask(self, question):
print("[?] "+question)
with patch_stdout():
try:
# new prompt instance, so the history / completer dont get mixed up
inp = await PromptSession().prompt_async(self.askPrompt,
bottom_toolbar = [None,self._bottom_toolbar][self.hasToolbar])
except KeyboardInterrupt:
return False
return inp
def runFromSync(self): def runFromSync(self):
asyncio.run(self.run()) asyncio.run(self.run())

View File

@ -1,2 +1,4 @@
- Refactor parsing logic
- Syntax information in toolbar (configurable) - Syntax information in toolbar (configurable)
- Allow other delimiters (e.g. ".")
- Support named arguments
- Include often used commands in a defaults collection (e.g. close)