Extend repl

This commit is contained in:
annihilator708 2018-05-23 00:09:21 +02:00
parent ad0044f601
commit 552fa1ba93

View File

@ -1,5 +1,7 @@
from discord.ext import commands from discord.ext import commands
import traceback import traceback
import discord
import inspect
import textwrap import textwrap
from contextlib import redirect_stdout from contextlib import redirect_stdout
import io import io
@ -45,6 +47,7 @@ class REPL:
if ctx.author.id not in self.bot.ownerlist: if ctx.author.id not in self.bot.ownerlist:
return await ctx.send('Only my contributors can use me like this :blush:', delete_after=10) return await ctx.send('Only my contributors can use me like this :blush:', delete_after=10)
if body is None: if body is None:
return await ctx.send( return await ctx.send(
'Please, use\n' 'Please, use\n'
@ -73,12 +76,13 @@ class REPL:
exec(to_compile, env) exec(to_compile, env)
except SyntaxError as e: except SyntaxError as e:
try: try:
msg = await ctx.send(f'```py\n{self.get_syntax_error(e)}\n```') await ctx.send(f'```py\n{self.get_syntax_error(e)}\n```')
except Exception as e: except Exception as e:
error = [self.get_syntax_error(e)[i:i+2000] for i in range(0, len(self.get_syntax_error(e)), 2000)] error = [self.get_syntax_error(e)[i:i + 2000] for i in
range(0, len(self.get_syntax_error(e)), 2000)]
for i in error: for i in error:
msg = await ctx.send(f'```py\n{i}\n```') await ctx.send(f'```py\n{i}\n```')
func = env['func'] func = env['func']
try: try:
@ -87,40 +91,141 @@ class REPL:
except Exception as e: except Exception as e:
value = stdout.getvalue() value = stdout.getvalue()
try: try:
msg = await ctx.send(f'```py\n{value}{traceback.format_exc()}\n```') await ctx.send(f'```py\n{value}{traceback.format_exc()}\n```')
except Exception as e: except Exception as e:
error = [value[i:i + 2000] for i in range(0, len(value), 2000)] error = [value[i:i + 2000] for i in range(0, len(value), 2000)]
for i in error: for i in error:
await ctx.send(f'```py\n{i}\n```') await ctx.send(f'```py\n{i}\n```')
tracebackerror = [traceback.format_exc()[i:i + 2000] for i in range(0, len(traceback.format_exc()), 2000)] tracebackerror = [traceback.format_exc()[i:i + 2000] for i in
range(0, len(traceback.format_exc()), 2000)]
for i in tracebackerror: for i in tracebackerror:
msg = await ctx.send(f'```py\n{i}\n```') await ctx.send(f'```py\n{i}\n```')
else: else:
value = stdout.getvalue() value = stdout.getvalue()
if ret is None: if ret is None:
if value: if value:
try: try:
msg = await ctx.send(f'```py\n{value}\n```') await ctx.send(f'```py\n{value}\n```')
except Exception as e: except Exception as e:
code = [value[i:i + 1980] for i in range(0, len(value), 1980)] code = [value[i:i + 1980] for i in range(0, len(value), 1980)]
for i in code: for i in code:
msg = await ctx.send(f'```py\n{i}\n```') await ctx.send(f'```py\n{i}\n```')
else: else:
self._last_result = ret self._last_result = ret
try: try:
code = [value[i:i + 1980] for i in range(0, len(value), 1980)] code = [value[i:i + 1980] for i in range(0, len(value), 1980)]
for i in code: for i in code:
msg = await ctx.send(f'```py\n{i}\n```') await ctx.send(f'```py\n{i}\n```')
except Exception as e: except Exception as e:
code = [value[i:i + 1980] for i in range(0, len(value), 1980)] code = [value[i:i + 1980] for i in range(0, len(value), 1980)]
for i in code: for i in code:
await ctx.send(f'```py\n{i}\n```') await ctx.send(f'```py\n{i}\n```')
modifyd_ret = [ret[i:i + 1980] for i in range(0, len(ret), 1980)] modifyd_ret = [ret[i:i + 1980] for i in range(0, len(ret), 1980)]
for i in modifyd_ret: for i in modifyd_ret:
msg = await ctx.send(f'```py\n{i}\n```') await ctx.send(f'```py\n{i}\n```')
@commands.command(hidden=True)
async def repl(self, ctx):
"""
Start a interactive python shell in chat.
Only the owner of this bot can use this command.
Usage:
- repl < python code >
Example:
- repl print(205554)
"""
if ctx.author.id not in self.bot.ownerlist:
return await ctx.send('Only my contributors can use me like this :blush:', delete_after=10)
msg = ctx.message
variables = {
'ctx': ctx,
'bot': self.bot,
'message': msg,
'server': msg.guild,
'channel': msg.channel,
'author': msg.author,
'_': None,
}
if msg.channel.id in self.sessions:
msg = await ctx.send('Already running a REPL session in this channel. Exit it with `quit`.')
self.sessions.add(msg.channel.id)
await ctx.send('Enter code to execute or evaluate. `exit()` or `quit` to exit.')
while True:
response = await self.bot.wait_for('message', check=lambda m: m.content.startswith(
'`') and m.author == ctx.author and m.channel == ctx.channel)
cleaned = self.cleanup_code(response.content)
if cleaned in ('quit', 'exit', 'exit()'):
msg = await ctx.send('Exiting.')
self.sessions.remove(msg.channel.id)
return
executor = exec
if cleaned.count('\n') == 0:
# single statement, potentially 'eval'
try:
code = compile(cleaned, '<repl session>', 'eval')
except SyntaxError:
pass
else:
executor = eval
if executor is exec:
try:
code = compile(cleaned, '<repl session>', 'exec')
except SyntaxError as e:
try:
await ctx.send(f'```Python\n{self.get_syntax_error(e)}\n```')
except Exception as e:
error = [self.get_syntax_error(e)[i:i + 2000] for i in
range(0, len(self.get_syntax_error(e)), 2000)]
for i in error:
await ctx.send(f'```Python\n{i}\n```')
variables['message'] = response
fmt = None
stdout = io.StringIO()
try:
with redirect_stdout(stdout):
result = executor(code, variables)
if inspect.isawaitable(result):
result = await result
except Exception as e:
value = stdout.getvalue()
await ctx.send(f'```Python\n{value}{traceback.format_exc()}\n```')
continue
else:
value = stdout.getvalue()
if result is not None:
fmt = '{}{}'.format(value, result)
variables['_'] = result
elif value:
fmt = value
try:
if fmt is not None:
if len(fmt) > 1980:
code = [fmt[i:i + 1980] for i in range(0, len(fmt), 1980)]
for i in code:
await ctx.send(f'```py\n{i}\n```')
else:
await ctx.send(fmt)
except discord.Forbidden:
pass
except discord.HTTPException as e:
await ctx.send(f'Unexpected error: `{e}`')
def setup(bot): def setup(bot):
bot.add_cog(REPL(bot)) bot.add_cog(REPL(bot))