Как оболочка django `python manage.py shell` держит командную строку открытой?

Когда вы запускаете python manage.py shell, (или shell_plus, если у вас есть расширение), скрипт импортирует некоторые вещи для вас, а затем открывает обычную оболочку python, чтобы вы могли попробовать команды, в отличие от запуска скрипта и его немедленного закрытия.

Я знаю, что можно использовать переключатель -i при вызове скрипта, чтобы сохранить командную строку открытой, например, python -i foo.py запустит содержимое foo.py и при этом сохранит оболочку открытой, а не сразу закроет ее.

Я предполагаю, что когда вы вызываете django's manage.py shell, он каким-то образом использует ту же функцию, чтобы заставить командную строку оставаться открытой и не закрываться немедленно.

Как это работает? Могу ли я добавить что-то в любой файл сценария, чтобы он вел себя так же (без использования переключателя -i)?

В версии 2.2.3 команда управления shell имеет три программы, которые можно запустить - ipython, bpython и python. Для первых двух вы можете запустить оболочки, используя python:

    def ipython(self, options):
        from IPython import start_ipython
        start_ipython(argv=[])

    def bpython(self, options):
        import bpython
        bpython.embed()

Для python django устанавливает readline, а затем использует code.interact для запуска интерпретатора python:

    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)
            # Enable tab completion on systems using libedit (e.g. macOS).
            # These lines are copied from Python's Lib/site.py.
            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")

        # 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']:
            for pythonrc in OrderedSet([os.environ.get("PYTHONSTARTUP"), os.path.expanduser('~/.pythonrc.py')]):
                if not pythonrc:
                    continue
                if not os.path.isfile(pythonrc):
                    continue
                with open(pythonrc) as handle:
                    pythonrc_code = handle.read()
                # Match the behavior of the cpython shell where an error in
                # PYTHONSTARTUP prints an exception and continues.
                try:
                    exec(compile(pythonrc_code, pythonrc, 'exec'), imported_objects)
                except Exception:
                    traceback.print_exc()

        code.interact(local=imported_objects)
Вернуться на верх