社区所有版块导航
Python
python开源   Django   Python   DjangoApp   pycharm  
DATA
docker   Elasticsearch  
aigc
aigc   chatgpt  
WEB开发
linux   MongoDB   Redis   DATABASE   NGINX   其他Web框架   web工具   zookeeper   tornado   NoSql   Bootstrap   js   peewee   Git   bottle   IE   MQ   Jquery  
机器学习
机器学习算法  
Python88.com
反馈   公告   社区推广  
产品
短视频  
印度
印度  
Py学习  »  Python

Python 测试驱动开发(附福利)

编程派 • 7 年前 • 637 次点击  

作者:虫师

来源:http://www.cnblogs.com/fnng/p/5598546.html

文中提到的《Test-Driven Development with Python》一书,这几天出了第二版。在线阅读地址:http://www.obeythetestinggoat.com/pages/book.html#toc,也可点击阅读原文查看,在线版的排版媲美印刷版。本书的风评还是不错的,建议大家看看。


最近在看一本书《Test-Driven Development with Python》,里面非常详细的介绍了如何一步一步通过测试驱动开发(TDD)的方式开发Web项目。刚好这本书中使用了我之前所了解的一些技术,Django、selenium、unittest等。所以,读下来受益匪浅。

我相信不少开发都写单元测试,不过,一般是先写功能代码,然后,再写单元测试用例,在编写单元测试用例的过程中,可能需要调整功能代码,从而使单元测试用例通过。但是TDD就特别要求先写测试用例,后写实现代码。这一开始确实有些难。

这里就选择一个简单的例子向各位介绍一下TDD的流程(套路)。

编写功能测试用例:

首先,编写功能测试用例,functional_tests.py

  1. from selenium import webdriver

  2. browser = webdriver.Firefox()

  3. browser.get("http://127.0.0.1:8000")

  4. assert "Django" in browser.title

你没看错,这就是由Selenium编写的功能测试代码。打开Firefox浏览器,并访问http://127.0.0.1:8000,通过assert 判断浏览器标题是否包含"Django"。

然后,运行该测试用例。

  1. D:pydj>python functional_tests.py

  2. Traceback (most recent call last):

  3.  File "functional_tests.py", line 6, in

  4.    assert "Django" in browser.title

  5. AssertionError

测试用例失败了,这是必然的,因为我们还没有创建被测试的项目。但,这同样也是我们想要的结果。TDD的套路就是通过编写功能代码,使测试用例通过。

创建项目:

接下来创建Django项目:

  D:pydj>django-admin startproject superlists

当前项目结构如下:

  1. ├─ functional_tests.py  

  2. └─ superlists  

  3.   ├─ manage.py  

  4.   └─ superlists  

  5.     ├─ __init__.py  

  6.     ├─ settings.py  

  7.     ├─ urls.py  

  8.     └─ wsgi.py

进入项目目录,启动项目:

  1. D:pydj> cd superlists

  2. D: pydjsuperlists>python manage.py runserver

  3. Performing system checks...

  4. System check identified no issues (0 silenced).

  5. You have unapplied migrations; your app may not work properly until they are applied.

  6. Run 'python manage.py migrate' to apply them.

  7. June 13, 2016 - 23:23:29

  8. Django version 1.9.7, using settings 'superlists.settings'

  9. Starting development server at http://127.0.0.1:8000/

  10. Quit the server with CTRL-BREAK.

再次运行功能测试用例,functional_tests.py

接下来继续编写功能测试用例。functional_tests.py

  1. #coding=utf-8

  2. from selenium import webdriver

  3. import unittest

  4. class NewVisitorTest(unittest.TestCase):

  5.    def setUp(self):

  6.        self.browser = webdriver.Firefox()

  7.        self.browser.implicitly_wait(3)

  8.    def tearDown(self):

  9.        self.browser.close()

  10.    def test_case_start_a_list_and_retrieve_it_later(self):

  11.        # 小明听说有一个很酷的在线代办事项应用

  12.        # 她去看了这个应用首页

  13.        self.browser.get("http://127.0.0.1:8000")

  14.        # 它注意到网页的标题和头部包含"To-Do"这个词语。

  15.        self.assertIn("To-Do", self.browser.title)

  16. if __name__ == '__main__':

  17.    unittest.main()

这里用到了unittest 单元测试框架。如果,你不懂Python的单元测试,建议读者去学习unittest。

执行测试用例:

  1. C:Python35python.exe D:/pydj/functional_tests.py

  2. F

  3. ======================================================================

  4. FAIL: test_case_start_a_list_and_retrieve_it_later (__main__.NewVisitorTest)

  5. ----------------------------------------------------------------------

  6. Traceback (most recent call last ):

  7.  File "D:/pydj/functional_tests.py", line 21, in test_case_start_a_list_and_retrieve_it_later

  8.    self.assertIn("To-Do", self.browser.title)

  9. AssertionError: 'To-Do' not found in 'Welcome to Django'

  10. ----------------------------------------------------------------------

  11. Ran 1 test in 3.491s

  12. FAILED (failures=1)

测试用例又在预料之内的失败了!先不要着急解决这个问题,把项目创建完成。

D:pydjsuperlists>python3 manage.py startapp lists

将functional_tests.py放到superlists项目目录下。

单元测试与功能测试的区别:

正如给事物所贴的众多标签一样,单元测试和功能测试之间的界线有时不那么清晰。不过,二者之间有个基本区别:功能测试站在用户的角度从外部测试应用,单元测试则站在程序员的角度从内部测试应用。

我遵从的 TDD 方法同时使用这两种类型测试应用。采用的工作流程大致如下。

(1) 先写功能测试,从用户的角度描述应用的新功能。

(2) 功能测试失败后,想办法编写代码让它通过(或者说至少让当前失败的测试通过)。此时,使用一个或多个单元测试定义希望代码实现的效果,保证为应用中的每一行代码

(3) 单元测试失败后,编写最少量的应用代码,刚好让单元测试通过。有时,要在第 2 步和第 3 步之间多次往复,直到我们觉得功能测试有一点进展为止。

(4) 然后,再次运行功能测试,看能否通过,或者有没有进展。这一步可能促使我们编写一些新的单元测试和代码等。

由此可以看出,这整个过程中,功能测试站在高层驱动开发,而单元测试则从低层驱动我们做些什么。

打开/lists/tests.py文件,编写单元测试。

  1. from django.test import TestCase

  2. # Create your tests here.

  3. class SmokeTest(TestCase):

  4.    def test_bad_moths(self):

  5.        self.assertEqual(1 + 1,2)

运行单元测试:

  1. D:pydjsuperlists>python3 manage.py test

  2. Creating test database for alias 'default'...

  3. .

  4. ----------------------------------------------------------------------

  5. Ran 1 test in 0.001s

  6. OK

  7. Destroying test database for alias 'default'...

  OK,说明单元测试没问题。接下来就要编写真正的单元测试了(/lists/tests.py)。

  1. from django.core.urlresolvers import resolve

  2. from django.test import TestCase

  3. from django.http import HttpRequest

  4. from lists.views import home_page

  5. class HomePageTest(TestCase):

  6.    def test_root_url_resolves_to_home_page_view(self):

  7.        found = resolve('/')

  8.        self.assertEqual(found.func, home_page)

  9.    def test_home_page_returns_correct_html(self):

  10.        request = HttpRequest()

  11.        response = home_page(request)

  12.        self.assertTrue(response.content.startswith(b''))

  13.        self.assertIn(b'', response.content)

  14.        self.assertTrue(response.content.endswith(b''))

第一个用例(testrooturlresolvestohomepage_view):

resolve 是 Django 内部使用的函数,用于解析 URL,并将其映射到相应的视图函数上。检查解析网站根路径" /" 时,是否能找到名为 home_page 的函数。

第二个用例(testhomepagereturnscorrect_html):

创建了一个 HttpRequest 对象,用户在浏览器中请求网页时, Django 看到的就是HttpRequest 对象。把这个 HttpRequest 对象传给 home_page 视图,得到响应。听说响应对象是 HttpResponse类的实例时,你应该不会觉得奇怪。接下来我们断定响应的 .content 属性(即发送给用户的 HTML)中有特定的内容。

assertTrue()希望响应以 标签开头,并在结尾处关闭该标签。注意, response.content 是原始字节,不是 Python 字符串,因此对比时要使用 b'' 句法。b是BYTE字符串

assertIn()希望响应中有一个 ')

运行单元测试使其通过:

  1. D:pydjsuperlists>python3 manage.py test

  2. Creating test database for alias 'default'...

  3. ..

  4. ----------------------------------------------------------------------

  5. Ran 2 tests in 0.002s

  6. OK

  7. Destroying test database for alias 'default'...

最后不要忘了,配置superlists/urls.py文件。

  1. urlpatterns = [

  2.    #url(r'^admin/', admin.site.urls),

  3.    url(r'^$', views.home_page),

  4. ]

最后的最后,启动服务:

  D:pydjsuperlists>python manage.py runserver

运行功能测试用例使其通过:

  1. C:UsersfnngjDesktopsuperlists>python3 functional_tests.py

  2. F

  3. ======================================================================

  4. FAIL: test_can_start_a_list_and_retrieve_it_later (__main__. NewVisitorTest)

  5. ----------------------------------------------------------------------

  6. Traceback (most recent call last):

  7.  File "functional_tests.py", line 21, in test_can_start_a_list_and_retrieve_it_later

  8.    self.fail('Finish the test!')

  9. AssertionError: Finish the test!

  10. ----------------------------------------------------------------------

  11. Ran 1 test in 7.070s

  12. FAILED (failures=1)

书的内容极其连贯,整本书学下来,相当于自己动手通过TDD的方式开发了一个项目。我这里有所删减。

感兴趣的可以买这本书的中文版来学习。《Python web开发:测试驱动方法》 这中文名翻译的。。。


题图:pexels,CC0 授权。

点击阅读原文,查看更多 Python 教程和资源。


今天看啥 - 高品质阅读平台
本文地址:http://www.jintiankansha.me/t/PqI4Nkh624
Python社区是高质量的Python/Django开发社区
本文地址:http://www.python88.com/topic/3422
 
637 次点击