2007-09-15 06:16:14 +08:00
|
|
|
import os
|
2016-12-13 08:36:08 +08:00
|
|
|
import select
|
|
|
|
import sys
|
2015-12-14 03:33:39 +08:00
|
|
|
import warnings
|
2013-12-30 18:12:40 +08:00
|
|
|
|
2017-01-13 02:12:21 +08:00
|
|
|
from django.core.management import BaseCommand, CommandError
|
2016-08-05 21:09:20 +08:00
|
|
|
from django.utils.datastructures import OrderedSet
|
2015-12-14 03:33:39 +08:00
|
|
|
from django.utils.deprecation import RemovedInDjango20Warning
|
2007-08-16 14:06:55 +08:00
|
|
|
|
2012-07-17 08:37:52 +08:00
|
|
|
|
2014-06-18 07:07:54 +08:00
|
|
|
class Command(BaseCommand):
|
2016-12-13 08:36:08 +08:00
|
|
|
help = (
|
|
|
|
"Runs a Python interactive interpreter. Tries to use IPython or "
|
|
|
|
"bpython, if one of them is available. Any standard input is executed "
|
|
|
|
"as code."
|
|
|
|
)
|
|
|
|
|
2014-01-20 10:45:21 +08:00
|
|
|
requires_system_checks = False
|
2015-12-14 03:33:39 +08:00
|
|
|
shells = ['ipython', 'bpython', 'python']
|
2014-06-07 04:39:33 +08:00
|
|
|
|
|
|
|
def add_arguments(self, parser):
|
2016-03-29 06:33:29 +08:00
|
|
|
parser.add_argument(
|
|
|
|
'--plain', action='store_true', dest='plain',
|
2015-12-14 03:33:39 +08:00
|
|
|
help='Tells Django to use plain Python, not IPython or bpython. '
|
2016-03-29 06:33:29 +08:00
|
|
|
'Deprecated, use the `-i python` or `--interface python` option instead.',
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'--no-startup', action='store_true', dest='no_startup',
|
|
|
|
help='When using plain Python, ignore the PYTHONSTARTUP environment variable and ~/.pythonrc.py script.',
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'-i', '--interface', choices=self.shells, dest='interface',
|
|
|
|
help='Specify an interactive interpreter interface. Available options: "ipython", "bpython", and "python"',
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'-c', '--command', dest='command',
|
|
|
|
help='Instead of opening an interactive shell, run a command as Django and exit.',
|
|
|
|
)
|
2007-08-16 14:06:55 +08:00
|
|
|
|
2017-01-10 01:02:43 +08:00
|
|
|
def ipython(self, options):
|
2013-07-27 05:32:56 +08:00
|
|
|
from IPython import start_ipython
|
|
|
|
start_ipython(argv=[])
|
|
|
|
|
2015-12-14 03:33:39 +08:00
|
|
|
def bpython(self, options):
|
2010-12-13 07:03:24 +08:00
|
|
|
import bpython
|
|
|
|
bpython.embed()
|
|
|
|
|
2015-12-14 03:33:39 +08:00
|
|
|
def python(self, options):
|
|
|
|
import code
|
|
|
|
# Set up a dictionary to serve as the environment for the shell, so
|
|
|
|
# that tab completion works on objects that are imported at runtime.
|
|
|
|
imported_objects = {}
|
|
|
|
try: # Try activating rlcompleter, because it's handy.
|
|
|
|
import readline
|
|
|
|
except ImportError:
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
# We don't have to wrap the following import in a 'try', because
|
|
|
|
# we already know 'readline' was imported successfully.
|
|
|
|
import rlcompleter
|
|
|
|
readline.set_completer(rlcompleter.Completer(imported_objects).complete)
|
2016-02-28 01:13:01 +08:00
|
|
|
# Enable tab completion on systems using libedit (e.g. Mac OSX).
|
|
|
|
# These lines are copied from Lib/site.py on Python 3.4.
|
|
|
|
readline_doc = getattr(readline, '__doc__', '')
|
|
|
|
if readline_doc is not None and 'libedit' in readline_doc:
|
|
|
|
readline.parse_and_bind("bind ^I rl_complete")
|
|
|
|
else:
|
|
|
|
readline.parse_and_bind("tab:complete")
|
2012-07-17 08:37:52 +08:00
|
|
|
|
2015-12-14 03:33:39 +08:00
|
|
|
# We want to honor both $PYTHONSTARTUP and .pythonrc.py, so follow system
|
|
|
|
# conventions and get $PYTHONSTARTUP first then .pythonrc.py.
|
|
|
|
if not options['no_startup']:
|
2016-08-05 21:09:20 +08:00
|
|
|
for pythonrc in OrderedSet([os.environ.get("PYTHONSTARTUP"), os.path.expanduser('~/.pythonrc.py')]):
|
2015-12-14 03:33:39 +08:00
|
|
|
if not pythonrc:
|
|
|
|
continue
|
|
|
|
if not os.path.isfile(pythonrc):
|
|
|
|
continue
|
|
|
|
try:
|
|
|
|
with open(pythonrc) as handle:
|
|
|
|
exec(compile(handle.read(), pythonrc, 'exec'), imported_objects)
|
|
|
|
except NameError:
|
|
|
|
pass
|
|
|
|
code.interact(local=imported_objects)
|
2010-12-13 07:03:03 +08:00
|
|
|
|
2014-06-18 07:07:54 +08:00
|
|
|
def handle(self, **options):
|
2015-12-14 03:33:39 +08:00
|
|
|
if options['plain']:
|
|
|
|
warnings.warn(
|
|
|
|
"The --plain option is deprecated in favor of the -i python or --interface python option.",
|
|
|
|
RemovedInDjango20Warning
|
|
|
|
)
|
|
|
|
options['interface'] = 'python'
|
2012-07-17 08:37:52 +08:00
|
|
|
|
2015-11-07 19:07:28 +08:00
|
|
|
# Execute the command and exit.
|
|
|
|
if options['command']:
|
|
|
|
exec(options['command'])
|
|
|
|
return
|
|
|
|
|
2016-12-13 08:36:08 +08:00
|
|
|
# Execute stdin if it has anything to read and exit.
|
|
|
|
# Not supported on Windows due to select.select() limitations.
|
|
|
|
if sys.platform != 'win32' and select.select([sys.stdin], [], [], 0)[0]:
|
|
|
|
exec(sys.stdin.read())
|
|
|
|
return
|
|
|
|
|
2015-12-14 03:33:39 +08:00
|
|
|
available_shells = [options['interface']] if options['interface'] else self.shells
|
|
|
|
|
|
|
|
for shell in available_shells:
|
|
|
|
try:
|
|
|
|
return getattr(self, shell)(options)
|
2007-08-16 14:06:55 +08:00
|
|
|
except ImportError:
|
|
|
|
pass
|
2017-01-13 02:12:21 +08:00
|
|
|
raise CommandError("Couldn't import {} interface.".format(shell))
|