Extend repl
This commit is contained in:
parent
ad0044f601
commit
552fa1ba93
125
src/cogs/code.py
125
src/cogs/code.py
@ -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))
|
||||||
Loading…
x
Reference in New Issue
Block a user