django单元测试历险记

简介:

最近需要测试一个服务器端的django项目,查看了一下django的文档,发现django为了更加方便的对web应用进行测试,提供了一些便捷的测试方法。并且,专门有一篇文档讲诉如何测试django应用。

http://docs.djangoproject.com/en/dev/topics/testing/ 

快速横扫了一下文档后,初步印象是django默认支持Doctests和Unit tests两个测试框架的,同时提供了一些辅助的测试类,比如Test Client、TestCase、Email Service。通过Client,可以方便的发起一个get或者post请求,并且取得返回结果。而TestCase也是对unittest.TestCase进行了进一步的封装,省去了很多重复要写的代码,比如定义一个self.client。Email Service提供了方便的邮件发送的方法。

使用Test Client有什么好处呢?

1. 它不需要Web服务器运行起来。

2. 它执行更加快速。

3. 它能非常方便的取到被测应用内部的东西。

好的,既然那么好的东西,赶紧试用一下。首先,我新建了一个测试工程testdjango,设置了一个简单的页面,通过访问"/test",会返回一个字符串"abc"。

#  urls.py
urlpatterns  =  patterns( '' ,
    (r
' ^test$ ' ' views.test ' ),
)
#  views.py
from  django.http  import  HttpResponse

def  test(request):
    
return  HttpResponse( ' abc ' )

 

接下来,编写一个测试案例:

复制代码
# tests.py
from
 django.test  import  TestCase

class  ViewTest(TestCase):
    
def  test(self):
        response 
=  self.client.get( ' /test ' )
        self.failUnlessEqual(
' abc ' , response.content)
复制代码

 

如何才能找到你的测试案例呢?文档中说,会加载settings里INSTALLED_APPS的模块,并且在该模块目录的models.py和tests.py中查找测试案例。因此,我在settings的INSTALLED_APPS里添加了'testdjango',testdjango目录中正好有tests.py。

接下来要运行测试案例,由于安装文档的说明, 运行所有INSTALLED_APPS里的测试案例只要运行:

python manage.py test

如果想指定运行,可以:

python manage.py test testdjango

 

OK,运行一下,非常不幸,出现了如下异常:

"You haven't set the DATABASE_ENGINE setting yet." 

数据库??测试案例为什么还需要设置数据库?回过头仔细看下文档,发现确实有test database一节,大概含义是会生成一个临时的测试数据库,名字是test_前缀,如果想改还可以设定TEST_DATABASE_NAME属性。如果是sqlite,则默认是:memory:方式,即记在内存的方式。为了简单起见,我设置一个sqlite的。。

复制代码
DATABASE_ENGINE  =   ' sqlite3 '             #  'postgresql_psycopg2', 'postgresql', 'mysql', 'sqlite3' or 'oracle'.
DATABASE_NAME  =   ' :memory: '               #  Or path to database file if using sqlite3.
DATABASE_USER  =   ''               #  Not used with sqlite3.
DATABASE_PASSWORD  =   ''           #  Not used with sqlite3.
DATABASE_HOST  =   ''               #  Set to empty string for localhost. Not used with sqlite3.
DATABASE_PORT  =   ''               #  Set to empty string for default. Not used with sqlite3.
复制代码

再次执行,看到输出窗口开始输出了:

Creating test database

突然又是一个异常:

" UnboundLocalError: local variable 'int_alias' referenced before assignment "

这个问题让我google了好一阵,在djangoproject里还专门有一个ticket报了这个问题,但最后确认为不是django的bug不了了之。

http://code.djangoproject.com/ticket/10930 

奇怪的是,只有在WingIDE中启动才会出现,在命令行里执行就不会有这个错误。后来看了一下WingIDE的设置,原来可以Debug里的Exception可以设置Never Report和Always Report,在Never Report里添加UnboundLocalError,同时在Always Report里去掉,这个异常就没有出现了。

再次运行,开始看到一大段的输出,时不时还有一些异常:

django.template.TemplateDoesNotExist: registration/password_change_form.html 

输出大致如下:

Code

原来运行的是manage.py test,默认把INSTALLED_APP里的其他模块的测试案例也运行了起来。这些异常也是允许那些测试案例抛出的,而我定义的测试案例貌似没有运行。来回看了几遍文档,百思不得其解。最后,打算直接看django的源码,为何没有加载我的测试案例,而像django.contrib.auth里的测试案例却都能执行起来。于是我一步一步的跟进去调试。最后发现:

首先会查找当前目录的models.py,如果这个模块Import失败的话不会再继续从tests模块里找。

也就是说,我必须定义一个models.py文件,即使里面没有内容。于是,我添加了一个models.py文件问题就解决了!

添加一个没什么用的models.py和设置数据库参数,看上去似乎对我来说没什么意义,但在某些情况下,这样的方式能够比较简单的解决之前的问题然后展开测试。 再细看文档时,其中还有提到的是,我们可以自己设置一个test runner,安装我们定义的方式去执行。听上去不错,这样我就可以不用去添加什么models.py和修改数据库设置了。django默认是会加载django.test.simple.run_tests函数,我们需要做的就是一个自己写一个run_tests函数,通过设置TEST_RUNNER属性,加载我们的test runner。

通过查看默认的test runner代码,大致了解了基本原理后,我对默认的test runner进行了大量的简化,去除了models.py的加载,去除了数据库的依赖。下面是我的test runner,代码很短:

 

复制代码
import  unittest
from  django.conf  import  settings
from  django.test.utils  import  setup_test_environment, teardown_test_environment
from  django.test.testcases  import  OutputChecker, DocTestRunner, TestCase

TEST_MODULE 
=   ' tests '

def  build_suite(label):
    suite 
=  unittest.TestSuite()
    
try :
        app_path 
=  label  +   ' . '   +  TEST_MODULE
        test_module 
=   __import__ (app_path, {}, {}, TEST_MODULE)
    
except  ImportError, e:
        test_module 
=  None
    
    
if  test_module:
        
if  hasattr(test_module,  ' suite ' ):
            suite.addTest(test_module.suite())
        
else :
            suite.addTest(unittest.defaultTestLoader.loadTestsFromModule(test_module))
    
return  suite

def  run_tests(test_labels, verbosity = 1 , interactive = True, extra_tests = []):
    setup_test_environment()

    settings.DEBUG 
=  False
    
    suite 
=  unittest.TestSuite()

    
if  test_labels:
        
for  label  in  test_labels:
            suite.addTest(build_suite(label))

    
for  test  in  extra_tests:
        suite.addTest(test)
        
    result 
=  unittest.TextTestRunner(verbosity = verbosity).run(suite)

    teardown_test_environment()

    
return  len(result.failures)  +  len(result.errors)
复制代码


不要忘记了设在TEST_RUNNER:

TEST_RUNNER   =   ' testdjango.testrunner.run_tests '

运行一下看看,发现我的测试案例失败了:

复制代码
======================================================================
ERROR: test (testdjango.tests.testviews.ViewTest)
----------------------------------------------------------------------
Traceback (most recent call last):
  File 
" D:\Python25\Lib\site-packages\django\test\testcases.py " , line  242 in   __call__
    self._pre_setup()
  File 
" D:\Python25\Lib\site-packages\django\test\testcases.py " , line  217 in  _pre_setup
    self._fixture_setup()
  File 
" D:\Python25\Lib\site-packages\django\test\testcases.py " , line  439 in  _fixture_setup
    
if   not  settings.DATABASE_SUPPORTS_TRANSACTIONS:
  File 
" D:\Python25\Lib\site-packages\django\utils\functional.py " , line  273 in   __getattr__
    
return  getattr(self._wrapped, name)
AttributeError: 
' Settings '  object has no attribute  ' DATABASE_SUPPORTS_TRANSACTIONS '

----------------------------------------------------------------------
Ran 0 tests 
in  0.000s

FAILED (errors
= 1 )
复制代码

 

为什么还需要依赖数据库??哦,原来我在测试案例中用到了本文开头提到的django封装后的TestCase,它的内部是有数据库相关的操作的。看来,要使用我这个test runner,就不能再使用django的TestCase了,而是使用unittest.TestCase了。因此,修改测试案例如下:

复制代码
import  unittest
from  django.test  import  Client

class  ViewTest(unittest.TestCase):
    
def  test(self):
        self.client 
=  Client()
        response 
=  self.client.get( ' /test ' )
        self.failUnlessEqual(response.status_code, 
200 )
        self.failUnlessEqual(
' abc ' , response.content)
复制代码

 

大功告成!输出结果:

----------------------------------------------------------------------
Ran 
1  test  in  0.937s

OK

 

因为是历险记,所有把很多过程的东西拿出来了。最后,把我最后能用的代码传一份上来,希望能够有些帮助,如果过程中有什么不对的地方,也请指出,谢谢!!

/Files/coderzh/testdjango.rar 

 

 

本文转自CoderZh博客园博客,原文链接:http://www.cnblogs.com/coderzh/archive/2009/11/15/1603315.html,如需转载请自行联系原作者

目录
打赏
0
0
0
0
19
分享
相关文章
Django测试入门:打造坚实代码基础的钥匙
Django测试入门:打造坚实代码基础的钥匙
69 3
django集成pytest进行自动化单元测试实战
在Django项目中集成Pytest进行单元测试可以提高测试的灵活性和效率,相比于Django自带的测试框架,Pytest提供了更为丰富和强大的测试功能。本文通过一个实际项目ishareblog介绍django集成pytest进行自动化单元测试实战。
118 3
django集成pytest进行自动化单元测试实战
Django中的单元测试
【8月更文挑战第11天】本文详述了如何运用Django框架内置的单元测试与集成测试工具来确保Web应用的代码质量。通过具体示例展示了单元测试的基础,如测试简单函数的正确性;以及集成测试的应用,验证应用组件间的协作无误。文中还强调了最佳实践,比如保持高测试覆盖率、确保测试独立且高效执行等。此外,还介绍了测试驱动开发(TDD)的方法,并讨论了模拟(mocking)技术在测试中的重要性。最后,提到了自动化测试与持续集成的实施方式,以及探索更多高级测试工具和技术的可能性,如行为驱动开发(BDD)、性能测试和静态代码分析等,全方位提升应用的稳定性和用户体验。
83 6
在django中的测试架构
【6月更文挑战第13天】该文主要讨论Django项目的测试数据查询和测试架构。文中展示了如何创建`TestCase`子类进行测试,并提供了执行测试的不同选项,如增加详细信息、并行运行和选择性运行特定测试。
64 2
在django中的测试架构
Django视图测试:构建可靠Web应用的关键步骤
Django视图测试:构建可靠Web应用的关键步骤
50 0
使用django构建表单测试
【6月更文挑战第14天】该文档介绍了如何对本地库进行自动化测试,特别是关注于代码结构和模型测试。作者鼓励为其他模型和表单创建类似的测试,并提及测试应避免对底层框架的重复验证。
86 0
使用django构建表单测试
Django自动化测试入门:单元测试与集成测试
【4月更文挑战第15天】本文介绍了Django的自动化测试,包括单元测试和集成测试。单元测试专注于单个视图、模型等组件的正确性,而集成测试则测试组件间的交互。Django测试框架提供`TestCase`和`Client`进行单元和集成测试。通过编写测试,开发者能确保代码质量、稳定性和应用的正确协同工作。运行测试使用`python manage.py test`命令,建议将其纳入日常开发流程。
Django的测试工具与框架:提升开发效率与质量
【4月更文挑战第15天】本文探讨了Django的测试工具和框架,包括单元测试、集成测试和功能测试,以及其基于unittest的测试框架特点,如易于集成、丰富断言和数据库支持。通过编写全面测试、采用TDD、自动化测试等方式,开发者能提升开发效率和代码质量。利用Django的测试支持对确保应用稳定性、用户体验及软件可维护性至关重要。
Python【算法中心 02】Web框架Django管理页面使用(管理员账号创建+API使用+应用添加)GreenPlum数据库引擎及API测试
Python【算法中心 02】Web框架Django管理页面使用(管理员账号创建+API使用+应用添加)GreenPlum数据库引擎及API测试
141 0
django drf 案例--实现url编码和json和dict格式转化小工具(涉及定义模型类,序列化器,类视图,路由),接口测试
django drf 案例--实现url编码和json和dict格式转化小工具(涉及定义模型类,序列化器,类视图,路由),接口测试

热门文章

最新文章

AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等