Проблемы с активацией виртуального сервера на сервере через Fabric

Я пытаюсь запустить некоторые команды управления Django через Fabric на моем промежуточном сервере.

Проблема заключается в том, что Fabric не может активировать virtualenv и, таким образом, используя системные python/libs при выполнении команд.

На сервере приложение Django запускается с использованием virtualenv (нет, я не использую virtualenvwrapper еще...)

Используя Fabric (1.0.1), команда может выглядеть так, когда вы запускаете из моего окна:

Метод fabfile:

def collectstatic():
    require('settings', provided_by=[production, staging])

    with settings(warn_only=True):
        run('source %(env_path)s/bin/activate && python %(repo_path)s/%(project_name)s/configs/%(settings)s/manage.py collectstatic --noinput -v0' % env)

Выход:

$ fab staging master collectstatic
[myserver.no] Executing task 'master'
[myserver.no] Executing task 'collectstatic'
[myserver.no] run: source /home/newsapps/sites/mysite/env/bin/activate && python /home/newsapps/sites/mysite/repository/mysite/configs/staging/manage.py collectstatic --noinput -v0
[myserver.no] Login password: 
[myserver.no] out: Unknown command: 'collectstatic'
[myserver.no] out: Type 'manage.py help' for usage.

Я знаю, конечно, что команда jango collectstatic не существует в версиях до 1.3, что заставляет med думать, что системный python (который использует Django 1.2) используется.

My fabfile/project layout основан на отличном fabfile от участников Tribapps

Итак, я создал метод ткани для проверки pythonversion:

def pythonver():
    require('settings', provided_by=[production, staging])

    with settings(warn_only=True):

    run('source %(env_path)s/bin/activate && echo "import sys; print sys.path" | python ' % env)

При запуске он дает следующий результат:

$ fab staging master pythonver
[myserver.no] Executing task 'master'
[myserver.no] Executing task 'pythonver'
[myserver.no] run: source /home/newsapps/sites/mysite/env/bin/activate && echo "import sys; print sys.path" | python 
[myserver.no] Login password: 
[myserver.no] out: ['', '/usr/lib/python2.6', '/usr/lib/python2.6/plat-linux2', '/usr/lib/python2.6/lib-tk', '/usr/lib/python2.6/lib-old', '/usr/lib/python2.6/lib-dynload', '/usr/lib/python2.6/dist-packages', '/usr/lib/pymodules/python2.6', '/usr/lib/pymodules/python2.6/gtk-2.0', 

Как вы можете видеть, он использует системный python, а не мой virtualenv, расположенный в home/newsapps/sites/mysite/env

Но если я запустил эту команду непосредственно на сервере

source /home/newsapps/sites/mysite/env/bin/activate && echo "import sys; print sys.path" | python 

.. затем он выдает правильные пути из virtualenv

Что я делаю неправильно, так как команды не запускаются с помощью python из моего virtualenv с помощью Fabric?

Ответ 1

Вы должны вызывать версию python из вашего каталога virtualenv bin, тогда вы будете уверены, что использует виртуальную версию python.

/home/newsapps/sites/mysite/env/bin/python /home/newsapps/sites/mysite/repository/mysite/configs/staging/manage.py collectstatic --noinput -v0

Ответ 2

Я бы не стал беспокоиться об активации virtualenv, просто дайте полный путь к интерпретатору python virtualenv. Затем будет использоваться правильный PYTHONPATH и т.д.

Ответ 3

У меня была та же проблема. Не удалось решить его простым способом. Поэтому я просто использовал полный путь к файлу bin python внутри virtualenv. Я не профессионал в Python, но, я думаю, это то же самое в конце. В моем файле fab это выглядит примерно так:

PYTHON = '/home/dudus/.virtualenvs/pai/bin/python'
PIP = '/home/dudus/.virtualenvs/pai/bin/pip'

def update_db():
    with cd(REMOTE_DIR + 'application/'):
        run('%s ./manage.py syncdb --settings="%s"' % 
            (PYTHON, SETTINGS)) # syncdb
        run('%s ./manage.py migrate --settings="%s"' % 
            (PYTHON, SETTINGS)) # south migrate

Ответ 4

Это будет отлично работать:)

from __future__ import with_statement
from fabric.api import *
from contextlib import contextmanager as _contextmanager

env.hosts = ['servername']
env.user = 'username' 
env.directory = '/path/to/virtualenvs/project' 
env.activate = 'source /path/to/virtualenvs/project/bin/activate'

@_contextmanager
def virtualenv():
    with cd(env.directory):
        with prefix(env.activate):
            yield

def deploy():
    with virtualenv():
        run('pip freeze')