2

changegroup フック中に、DB でいくつかのビルド要求をキューに入れようとしています。DB テーブルは Django アプリによって管理されます。そのため、フックでは、Django モデルを使用してビルド リクエストを送信したいと考えています。ただし、モデルを正常にインポートするための設定を正しく行うことができません。

この ( https://docs.djangoproject.com/en/1.3/ref/django-admin/ ) 情報に基づいて、django サイトを sys パスに追加し、DJANGO_SETTINGS_MODULE を設定するだけです。django サイトは /opt/mysite にあるため、Python コンソールからこれを試してみます。

>>> import sys
>>> import os
>>> sys.path.append('/opt')
>>> os.environ['DJANGO_SETTINGS_MODULE'] = 'mysite.settings'
>>> from mysite.myapp.models import Build
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "/opt/mysite/myapp/models.py", line 1, in <module>
    from django.db import models
  File "/usr/lib/pymodules/python2.7/django/db/__init__.py", line 78, in <module>
    connection = connections[DEFAULT_DB_ALIAS]
  File "/usr/lib/pymodules/python2.7/django/db/utils.py", line 93, in __getitem__
    backend = load_backend(db['ENGINE'])
  File "/usr/lib/pymodules/python2.7/django/db/utils.py", line 33, in load_backend
    return import_module('.base', backend_name)
  File "/usr/lib/pymodules/python2.7/django/utils/importlib.py", line 35, in import_module
    __import__(name)
  File "/usr/lib/pymodules/python2.7/django/db/backends/mysql/base.py", line 28, in <module>
    from django.db import utils
ImportError: cannot import name utils
>>>

2 回目の試行:

これに基づいて ( http://www.b-list.org/weblog/2007/sep/22/standalone-django-scripts/ ) setup_environ() を使用できます。Python シェルでこれを試すと、動作します:

>>> import sys
>>> sys.path.append('/opt')
>>> from django.core.management import setup_environ
>>> from mysite import settings
>>> setup_environ(settings)
'/opt/mysite'
>>> from mysite.myapp.models import Build
>>> Build.objects.all()
[<Build: #1 F nkj sdfsdfsdfs43qg test_branch>, <Build: #2 F nkj sdfsdfsdfs43qg test_branch>, <Build: #13 Q   >, <Build: #14 Q   nkj_bug243>, <Build: #11 F nkj 444hwe45hedrrt nkj_bug272>, <Build: #12 F nkj sdfsdfsdfs43qg test_branch>]

ただし、Mercurial フックでこれを試すと、Build モデルをインポートしようとするとすぐにエラーが発生します。

#!/usr/bin/env python
import sys
path = '/opt'
if path not in sys.path:
    sys.path.append(path)

from django.core.management import setup_environ
from mysite import settings
setup_environ(settings)

from mysite.myapp.models import Build
import datetime

... do stuff ...

「from mysite.myapp.models import Build」行に到達すると、次のエラー出力が表示されます。

** unknown exception encountered, please report by visiting
**  http://mercurial.selenic.com/wiki/BugTracker
** Python 2.7.2+ (default, Oct  4 2011, 20:06:09) [GCC 4.6.1]
** Mercurial Distributed SCM (version 1.9.1)
** Extensions loaded:
Traceback (most recent call last):
  File "/usr/bin/hg", line 38, in <module>
    mercurial.dispatch.run()
  File "/usr/lib/python2.7/dist-packages/mercurial/dispatch.py", line 27, in run
    sys.exit(dispatch(request(sys.argv[1:])))
  File "/usr/lib/python2.7/dist-packages/mercurial/dispatch.py", line 64, in dispatch
    return _runcatch(req)
  File "/usr/lib/python2.7/dist-packages/mercurial/dispatch.py", line 87, in _runcatch
    return _dispatch(req)
  File "/usr/lib/python2.7/dist-packages/mercurial/dispatch.py", line 679, in _dispatch
    cmdpats, cmdoptions)
  File "/usr/lib/python2.7/dist-packages/mercurial/dispatch.py", line 454, in runcommand
    ret = _runcommand(ui, options, cmd, d)
  File "/usr/lib/python2.7/dist-packages/mercurial/dispatch.py", line 733, in _runcommand
    return checkargs()
  File "/usr/lib/python2.7/dist-packages/mercurial/dispatch.py", line 687, in checkargs
return cmdfunc()
  File "/usr/lib/python2.7/dist-packages/mercurial/dispatch.py", line 676, in <lambda>
    d = lambda: util.checksignature(func)(ui, *args, **cmdoptions)
  File "/usr/lib/python2.7/dist-packages/mercurial/util.py", line 385, in check
return func(*args, **kwargs)
  File "/usr/lib/python2.7/dist-packages/mercurial/commands.py", line 3884, in push
    newbranch=opts.get('new_branch'))
  File "/usr/lib/python2.7/dist-packages/mercurial/localrepo.py", line 1428, in push
lock=lock)
  File "/usr/lib/python2.7/dist-packages/mercurial/localrepo.py", line 1849, in addchangegroup
source=srctype, url=url)
  File "/usr/lib/python2.7/dist-packages/mercurial/localrepo.py", line 224, in hook
    return hook.hook(self.ui, self, name, throw, **args)
  File "/usr/lib/python2.7/dist-packages/mercurial/hook.py", line 160, in hook
    mod = extensions.loadpath(path, 'hghook.%s' % hname)
  File "/usr/lib/python2.7/dist-packages/mercurial/extensions.py", line 45, in loadpath
    return imp.load_source(module_name, path)
  File "/opt/unapse/belvedere/mercurial_hook.py", line 14, in <module>
    from unapse.belvedere.models import Build
  File "/usr/lib/python2.7/dist-packages/mercurial/demandimport.py", line 109, in _demandimport
    mod = _origimport(name, globals, locals)
  File "/opt/unapse/belvedere/models.py", line 1, in <module>
    from django.db import models
  File "/usr/lib/python2.7/dist-packages/mercurial/demandimport.py", line 109, in _demandimport
    mod = _origimport(name, globals, locals)
  File "/usr/lib/pymodules/python2.7/django/db/__init__.py", line 14, in <module>
    if not settings.DATABASES:
  File "/usr/lib/pymodules/python2.7/django/utils/functional.py", line 276, in __getattr__
    self._setup()
  File "/usr/lib/pymodules/python2.7/django/conf/__init__.py", line 42, in _setup
    self._wrapped = Settings(settings_module)
  File "/usr/lib/pymodules/python2.7/django/conf/__init__.py", line 139, in __init__
    logging_config_func(self.LOGGING)
  File "/usr/lib/python2.7/logging/config.py", line 776, in dictConfig
    dictConfigClass(config).configure()
  File "/usr/lib/python2.7/logging/config.py", line 575, in configure
    '%r: %s' % (name, e))
ValueError: Unable to configure handler 'mail_admins': __import__() argument 1 must be string, not DictConfigurator

この問題の原因について何か考えはありますか?

4

2 に答える 2

1

DictConfigurator エラーは Python のバグであり、次の 2.7 リリース ( http://bugs.python.org/issue12718 ) で修正されることを願っています。何か (おそらく水銀) がインポートをオーバーライドしており、その場合、Python のログ コードは失敗します。

それまでは、これを修正するために私がしたことです。django/utils から dictconfig.py を Python システム パスのどこかにコピーします (フックと一緒に配置します)。行を変更する

importer = __import__

に:

def importer(self, *args):
    return __import(*args)__

そして、あなたのdjangoプロジェクトのsettings.pyで、最後に追加します:

LOGGING_CONFIG = 'website.dictconfig.dictConfig'

website.dictconfig を dictconfig ファイルを配置したパスに更新して、python が見つけられるようにします。

更新:もっと簡単な方法があると思います。他のものをインポートする前に、これをフック スクリプトの先頭に追加します。

from mercurial import demandimport;
demandimport.disable()

これにより、Mercurial はインポートのオーバーライドを元に戻し、問題をきれいに解決するように見えます。

于 2013-03-28T19:31:08.913 に答える
0

環境を virtualenv dir で実行している場合は、それをサイト パッケージに追加する必要があります。また、django dir をsys.path末尾ではなく先頭に配置することをお勧めします。

Django アプリを他の Python アプリに統合するために使用するコードを次に示します。

import os
import sys

# point this at the virtualenv dir that your django deployment runs out of
# you are using virtualenv. right?
DJANGO_ENV = '/home/core/python-envs/production'

# where your code lives (ie where settings.py is)
DJANGO_DIR = '/home/core/code/production'

# add our site packages
import site
site_packages = os.path.join(DJANGO_ENV, 'lib', 'python%s' % sys.version[:3], 'site-packages')
site.addsitedir(site_packages)

# put the main Django directory on first
sys.path.insert(0, DJANGO_DIR)

# setup the Django environment pointing to our settings
import settings
import django.core.management
django.core.management.setup_environ(settings)

# finally, import our objects
from yourapp.models import YourModel

最後に注意すべきことは、コードを一部のアプリケーションにインポートするときに、スコープの問題とインポートの衝突に遭遇したことです。これを回避するには、実行したい作業を関数でラップして、独自のスコープを取得します。

たとえば、次の代わりに:

from yourapp.models import YourModel
objs = YourModel.objects.filter(something=True)
...

次のようにします。

def do_work():
    from yourapp.models import YourModel
    objs = YourModel.objects.filter(something=True)
    ...
do_work()
于 2012-04-05T18:02:46.997 に答える