我可以在不将此应用程序合并到项目中的情况下为我的可重用 Django 应用程序启动测试吗?
我的应用使用了一些模型,所以需要提供(TEST_)DATABASE_*
设置。我应该在哪里存储它们以及我应该如何启动测试?
对于 Django 项目,我可以使用manage.py test
;运行测试。当我使用django-admin.py test
我的独立应用程序时,我得到:
错误:无法导入设置,因为环境变量 DJANGO_SETTINGS_MODULE 未定义。
这里有哪些最佳实践?
我可以在不将此应用程序合并到项目中的情况下为我的可重用 Django 应用程序启动测试吗?
我的应用使用了一些模型,所以需要提供(TEST_)DATABASE_*
设置。我应该在哪里存储它们以及我应该如何启动测试?
对于 Django 项目,我可以使用manage.py test
;运行测试。当我使用django-admin.py test
我的独立应用程序时,我得到:
错误:无法导入设置,因为环境变量 DJANGO_SETTINGS_MODULE 未定义。
这里有哪些最佳实践?
Django (>= 1.4) test runner 的正确用法如下:
import django, sys
from django.conf import settings
settings.configure(DEBUG=True,
DATABASES={
'default': {
'ENGINE': 'django.db.backends.sqlite3',
}
},
ROOT_URLCONF='myapp.urls',
INSTALLED_APPS=('django.contrib.auth',
'django.contrib.contenttypes',
'django.contrib.sessions',
'django.contrib.admin',
'myapp',))
try:
# Django < 1.8
from django.test.simple import DjangoTestSuiteRunner
test_runner = DjangoTestSuiteRunner(verbosity=1)
except ImportError:
# Django >= 1.8
django.setup()
from django.test.runner import DiscoverRunner
test_runner = DiscoverRunner(verbosity=1)
failures = test_runner.run_tests(['myapp'])
if failures:
sys.exit(failures)
DjangoTestSuiteRunner 和 DiscoverRunner 具有大部分兼容的接口。
有关更多信息,您应该查阅“定义测试运行器”文档:
我以这样的解决方案结束(它的灵感来自 django-voting 中的解决方案):
创建文件,例如。测试目录中的“runtests.py”包含:
import os, sys
from django.conf import settings
DIRNAME = os.path.dirname(__file__)
settings.configure(DEBUG = True,
DATABASE_ENGINE = 'sqlite3',
DATABASE_NAME = os.path.join(DIRNAME, 'database.db'),
INSTALLED_APPS = ('django.contrib.auth',
'django.contrib.contenttypes',
'django.contrib.sessions',
'django.contrib.admin',
'myapp',
'myapp.tests',))
from django.test.simple import run_tests
failures = run_tests(['myapp',], verbosity=1)
if failures:
sys.exit(failures)
它允许通过python runtests.py
命令运行测试。它不需要安装依赖项(例如构建),并且不会损害将应用程序合并到更大项目中时运行的测试。
对于 Django 1.7,它略有不同。假设您有以下 app 目录结构foo
:
foo
|── docs
|── foo
│ ├── __init__.py
│ ├── models.py
│ ├── urls.py
│ └── views.py
└── tests
├── foo_models
│ ├── __init__.py
│ ├── ...
│ └── tests.py
├── foo_views
│ ├── __init__.py
│ ├── ...
│ └── tests.py
├── runtests.py
└── urls.py
这就是 Django 项目本身构建其测试的方式。
您想foo/tests/
使用以下命令运行所有测试:
python3 runtests.py
您还希望能够从 的父目录运行命令tests
,例如通过 Tox 或 Invoke,就像python3 foo/tests/runtests.py
.
我在这里介绍的解决方案是可重复使用的,只需调整应用程序的名称foo
(必要时还需要调整其他应用程序)。它们不能通过modify_settings安装,因为它会错过数据库设置。
需要以下文件:
网址.py
"""
This urlconf exists because Django expects ROOT_URLCONF to exist. URLs
should be added within the test folders, and use TestCase.urls to set them.
This helps the tests remain isolated.
"""
urlpatterns = []
运行测试.py
#!/usr/bin/env python3
import glob
import os
import sys
import django
from django.conf import settings
from django.core.management import execute_from_command_line
BASE_DIR = os.path.abspath(os.path.dirname(__file__))
sys.path.append(os.path.abspath(os.path.join(BASE_DIR, '..')))
# Unfortunately, apps can not be installed via ``modify_settings``
# decorator, because it would miss the database setup.
CUSTOM_INSTALLED_APPS = (
'foo',
'django.contrib.admin',
)
ALWAYS_INSTALLED_APPS = (
'django.contrib.auth',
'django.contrib.contenttypes',
'django.contrib.sessions',
'django.contrib.messages',
'django.contrib.staticfiles',
)
ALWAYS_MIDDLEWARE_CLASSES = (
'django.contrib.sessions.middleware.SessionMiddleware',
'django.middleware.common.CommonMiddleware',
'django.middleware.csrf.CsrfViewMiddleware',
'django.contrib.auth.middleware.AuthenticationMiddleware',
'django.contrib.messages.middleware.MessageMiddleware',
'django.middleware.clickjacking.XFrameOptionsMiddleware',
)
settings.configure(
SECRET_KEY="django_tests_secret_key",
DEBUG=False,
TEMPLATE_DEBUG=False,
ALLOWED_HOSTS=[],
INSTALLED_APPS=ALWAYS_INSTALLED_APPS + CUSTOM_INSTALLED_APPS,
MIDDLEWARE_CLASSES=ALWAYS_MIDDLEWARE_CLASSES,
ROOT_URLCONF='tests.urls',
DATABASES={
'default': {
'ENGINE': 'django.db.backends.sqlite3',
}
},
LANGUAGE_CODE='en-us',
TIME_ZONE='UTC',
USE_I18N=True,
USE_L10N=True,
USE_TZ=True,
STATIC_URL='/static/',
# Use a fast hasher to speed up tests.
PASSWORD_HASHERS=(
'django.contrib.auth.hashers.MD5PasswordHasher',
),
FIXTURE_DIRS=glob.glob(BASE_DIR + '/' + '*/fixtures/')
)
django.setup()
args = [sys.argv[0], 'test']
# Current module (``tests``) and its submodules.
test_cases = '.'
# Allow accessing test options from the command line.
offset = 1
try:
sys.argv[1]
except IndexError:
pass
else:
option = sys.argv[1].startswith('-')
if not option:
test_cases = sys.argv[1]
offset = 2
args.append(test_cases)
# ``verbosity`` can be overwritten from command line.
args.append('--verbosity=2')
args.extend(sys.argv[offset:])
execute_from_command_line(args)
一些设置是可选的;他们提高了速度或更现实的环境。
第二个参数指向当前目录。它利用提供目录路径的特性来发现该目录下的测试。
对于我的可重用应用程序(django-moderation),我使用 buildout。我创建example_project
,我将它与 buildout 一起使用以在其上运行测试。我只是将我的应用程序放在example_project
.
当我想安装我的项目使用的所有依赖项并运行测试时,我只需要执行以下操作:
运行构建:
垃圾箱/扩建
运行 Django 1.1 和 Django 1.2 的测试:
bin/test-1.1 bin/test-1.2
在这里您可以找到如何配置可重用应用程序以使用 buildout 进行部署和测试运行的教程:http: //jacobian.org/writing/django-apps-with-buildout/
在这里,您将找到我在项目中使用的示例构建配置:
http://github.com/dominno/django-moderation/blob/master//buildout.cfg
在纯粹的pytest上下文中,为了提供足够的 Django 环境来让我的可重用应用程序在没有实际 Django 项目的情况下运行测试,我需要以下几部分:
pytest.ini:
[pytest]
DJANGO_SETTINGS_MODULE = test_settings
python_files = tests.py test_*.py *_tests.py
测试设置.py:
# You may need more or less than what's shown here - this is a skeleton:
DATABASES = {
'default': {
'ENGINE': 'django.db.backends.sqlite3',
}
}
INSTALLED_APPS = (
'django.contrib.admin',
'django.contrib.auth',
'django.contrib.contenttypes',
'django.contrib.messages',
'django.contrib.sessions',
'django.contrib.sites',
'django.contrib.staticfiles',
'todo',
)
ROOT_URLCONF = 'base_urls'
TEMPLATES = [
{
'DIRS': ['path/to/your/templates'), ],
}
]
MIDDLEWARE = [
'django.middleware.security.SecurityMiddleware',
'django.contrib.sessions.middleware.SessionMiddleware',
'django.middleware.common.CommonMiddleware',
'django.middleware.csrf.CsrfViewMiddleware',
'django.contrib.auth.middleware.AuthenticationMiddleware',
'django.contrib.messages.middleware.MessageMiddleware',
'django.middleware.clickjacking.XFrameOptionsMiddleware',
]
base_urls.py:
"""
This urlconf exists so we can run tests without an actual
Django project (Django expects ROOT_URLCONF to exist.)
It is not used by installed instances of this app.
"""
from django.urls import include, path
urlpatterns = [
path('foo/', include('myapp.urls')),
]
模板/base.html:
如果您的任何测试命中实际视图,则您的应用程序模板可能会扩展项目base.html
,因此该文件必须存在。就我而言,我刚刚创建了一个空文件templates/base.html
。
我现在可以pytest -x -v
从独立的可重用应用程序目录运行,而无需 Django 项目。
我知道这是一个旧线程,但我发现如果您的可重用应用程序有一个文件“tests.py”,那么您可以在项目文件夹中创建一个tests.py,然后编写以下内容......
from reusableapp.tests import *
然后,当您调用“python manage.py test”时,Django 将运行这些测试