django单元测试
django 单元测试小结
django 测试
从前很少写单元测试了,特别是web应用。最近不知不觉喜欢起来这个事情了,发现单元测试对于软件的模块,正交性有很大促进作用,因为函数,模块写的不合理,单元测试写起来就麻烦的多呀。公司的项目一直都是用Django,所以写点django单元测试的小总结,记录为主,备查。
测试的场景
框架Django1.8 测试工具 unittest, 要记得给test设置一个独特的settings。
测试请求 也就是测试整个view部分 官方案例 其中可能会遇到登录,或者时session怎么模拟的问题
测试带有orm的模块
- 需要mock的测试,比较多的情况是有第三方API调用, 发邮件,发短信这种
unittest提供的断言种类挺多,但是经常用的也就几个 self.assertContains, self.assertEqual, self.assertTrue
顺便提下有用的选项(我这里是单独给测试写了一个settings), 为了提高测试速度,可以把用不到的中间件,installed_apps之类的多余配置给去掉。
测试全部用例python manage.py test --setting settings_test测试某个APPpython manage.py test appname --setting settings_test测试某个app下的TeseCase类python manage.py test alarm.tests.ModelTestCase--setting settings_test-v {1,2,3}数字越大,显示的输出越详细,测试的日志信息python manage.py test --setting settings_test -v3其他的选项请查看--helppython manage.py test --help
用请求测试 views函数
DJANGO中提供了Client类来模拟http请求,可以模拟不同的method,然后就是请求参数的模拟,用起来很方面。
#coding:utf-8from django.test importTestCase,Clientfrom sendviews import*from core.tests import create_userclassSendviewsTestCase(TestCase):def setUp(self):self.user = create_user()self.device =Device(hostname="CN-BJ-0000-00",mac="ff:ff:ff:ff:ff:ff", user=self.user).save()def test_creat_sms(self):c =Client()rep = c.post("/acquireportal/createsms",{"phone":"13988902345","ssid":"erya","dmac":"ff:ff:ff:ff:ff:ff"})# 测试http请求的返回码是否正确self.assertEqual(rep.status_code,200)# 测试response的内容是否包含字符串self.assertContains(rep,"OK")# 测试response的内容是否包含字符串 方法二self.assertTrue('OK'in rep.content)
- 操作 session,例如用户登录,特殊的session值
from django.test importClientdef init_client(user):client =Client()client.login(username=user.username, password="lzz")s = client.sessions['cur_user_id']= user.ids.save()return client
- 增加header
from django.test.utils import setup_test_environmentsetup_test_environment()from django.test.client importClientc =Client()# get 请求,带参数,并增加headerc.get('/some/path/',{'qs_param':'foo'},**{'HTTP_USER_AGENT':'silly-human','REMOTE_ADDR':'127.0.0.1'})#get 请求,没有带参数,自定义headersc.get('/some/path/',**{'HTTP_USER_AGENT':'silly-human','REMOTE_ADDR':'127.0.0.1'})
- 使用 RequestFactory 对象来进行测试,不是从 http client来发起,某些情况会用到
from django.test importTestCase,RequestFactoryfrom django.http importHttpResponsefrom util.sign import generate_sign, validate_signfrom util.decorators import apiauth_required, SIGN_KEY@apiauth_required()def simpleapi(request):returnHttpResponse('ok')classDecoratorsTestCase(TestCase):def setUp(self):self.factory =RequestFactory()def test_apiauth(self):# create request objectkey = SIGN_KEYquery_string ={u"name": u"lzz", u"age": u"20", u"data": u"[python, java, golang, lua]"}token = generate_sign(query_string, key)query_string.update({u"sign": token})req =self.factory.post("/api/test", data=query_string)response = simpleapi(req)self.assertEqual(response.status_code,200)
- HTML 文本测试,使用 constants 来判断并不是个好的选择,可以用render之后的字符串对比。
对于需要登陆的view,有client也比较容易操作,还有一些特殊的session的检测等, 我这里做了一个简单的封装
from django.test importClientdef init_client(user):client =Client()client.login(username=user.username, password="lzz")s = client.sessions['cur_user_id']= user.ids.save()return client
带有mock的测试
对模块中的方法mock或者是对一个对象中的方法进行mock。真对测试函数中一些无法直接测试的函数设置默认的返回值, py3标准库中已经有了mock模块,py2需要自己安装, 推荐教程 使用Pyhton Mock进行单元测试1。 下面是个实际的代码片段。
import mockfrom django.test importTestCasefrom core.models importTenantfrom alarm.models import*from.controler importTenantAlarmclassModelTestCase(TestCase):def setUp(self):self.tenant =Tenant.objects.create(domainname="erya", comname=u"尔雅")@mock.patch.object(TenantAlarm,"sendAlarm")def test_record_alarm(self, mock_method):# record_alarm 这个中会调用sendAlarm方法mock_method.return_value =Nonecontent ="ccccc"atype =0rec_uid =0Alarm().record_alarm(content=content, atype=0,rec_tid=self.tenant.id)classTenantAlarmTestCase(TestCase):def setUp(self):self.tenant =Tenant.objects.create(domainname="erya", comname=u"尔雅")@mock.patch.object(TenantAlarm,"sendSMS", return_value=None)@mock.patch.object(TenantAlarm,"sendEmail", return_value=None)def test_send_alarm(self, method1, method2):content = u"报警了"ta =TenantAlarm(self.tenant.id, content,{u'SMS':0, u'EMAIL':0})ta.sendAlarm()@mock.patch('util.sendsms_com.send', return_value=1)def test_sendsms(self, send):ta =TenantAlarm(self.tenant.id,self.content,{u'SMS':0, u'EMAIL':0})ta.sendSMS()self.assertEqual(0,Account.objects.get(tenant=self.tenant).sms_num)self.account.sms_num =100self.account.save()ta.sendSMS()self.assertEqual(99,Account.objects.get(tenant=self.tenant).sms_num)
coverage
coverage是一个检查单元测试覆盖率的工具,django的文档中也有简要的说明coverage的集成 文档地址
#测试并收集测试信息coverage run --source='.' manage.py test --setting mandela.settings_test#查看测试结果coverage report -mNameStmtsMissCoverMissing----------------------------------------------------------------------------------------acquireportal/__init__.py 00100%acquireportal/controler.py 654728%22-56,60-71,76-79acquireportal/migrations/0001_initial.py 60100%acquireportal/migrations/0002_auto_20160622_1059.py 60100%acquireportal/migrations/0003_auto_20160622_1100.py 50100%....----------------------------------------------------------------------------------------TOTAL 8013585827%
覆盖率挺低的
django单元测试的更多相关文章
- [Django] 单元测试小记
从前很少写单元测试了,特别是web应用.最近不知不觉喜欢起来这个事情了,发现单元测试对于软件的模块,正交性有很大促进作用,因为函数,模块写的不合理,单元测试写起来就麻烦的多呀.公司的项目一直都是用Dj ...
- 如何进行Django单元测试
如何进行Django单元测试 Django的单元测试使用python的unittest模块,这个模块使用基于类的方法来定义测试.类名为django.test.TestCase,继承于python的un ...
- Django 单元测试
mock 测试 mock 是辅助单元测试的模块,用于测试不方便调用的别人的接口.举个简单的例子,比如说,我们测试django 写的微信登录接口,正常流程下,我们需要前端拉起授权窗口,获取jscode或 ...
- Django单元测试 相关知识
前言 本文,旨在说明python Django如何编写单元测试,从“背景”,“测试要求”,“代码编写”,“如何运行”,“检验测试覆盖度” 这几个方面来说明附上django的官方文档单元测试章节=> ...
- Django 单元测试笔记
引言 关于单元测试的基本知识这里不再讲述,简单一句话:单元测试是用一段代码去测试另一段代码.最常用的框架是unittest,这是python的单元测试框架,而django单元测试框架test.Test ...
- Django单元测试(一)
Django测试框架非常简单,首选方法是使用python标准库中的unittest模块. Writing tests Django的单元测试使用python的unittest模块,这个模块使用基于类的 ...
- django 单元测试错误总结
TestCase django自带有一个TestCase模块来进行测试,我们可以参考官网 来写单元测试的代码.我这里主要是总结一些错误. 用户无法登陆 我们有些api登录后才可以进行测试,所以我们可以 ...
- django 单元测试小结
测试的场景 框架Django1.8 测试工具 unittest, 要记得给test设置一个独特的settings. 测试请求 也就是测试整个view部分 官方案例 其中可能会遇到登录,或者时sessi ...
- Django单元测试中Fixtures用法
在使用单元测试时,有时候需要测试数据库中有数据,这时我们可以使用Django的Fixtures来生成测试数据. 基础配置 在settings.py 中配置如下内容: FIXTURE_DIRS = (' ...
随机推荐
- 串口 COM口 TTL RS-232 RS-485 区别 释疑
Point: 1.串口.COM口是指的物理接口形式(硬件).而TTL.RS-232.RS-485是指的电平标准(电信号). 2.接设备的时候,一般只接GND RX TX.不会接Vcc或者+3.3v的电 ...
- OGC学习课程
1.引言 由于项目需要,需要学习OGC相关地图标准,包括WMS.WFS.GML.SLD等,只是国内相关书籍大家都懂的,特向Google大师请教,得一秘籍<Open Web Mapping> ...
- 关于Jquery中ajax介绍
jQuery 底层 AJAX 实现.简单易用的高层实现见 $.get, $.post 等.$.ajax() 返回其创建的 XMLHttpRequest 对象.大多数情况下你无需直接操作该函数,除非你需 ...
- SQL超过锁请求
---超过锁请求 http://blog.sina.com.cn/s/blog_7fb3b1840100u4dj.html 1.查询造成死锁的SQL语句 当SQL数据库的监控报警显示死锁进程数过多时, ...
- SSH与Webservice整合记录
一.首先搭好SSH框架: 1. Struts:MyEclipse菜单栏MyEclipse——>Project Capabilities——>Add Struts Capabilities, ...
- iOS 解压打包静态库命令
p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 12.0px "Hannotate SC" } p.p2 { margin: 0.0px ...
- .net如何向csv添加一列
using System;using System.Collections.Generic;using System.Drawing;using System.Globalization;using ...
- SQL Server中Rowcount与@@Rowcount的用法
rowcount的用法: rowcount的作用就是用来限定后面的sql在返回指定的行数之后便停止处理,比如下面的示例, set rowcount 10select * from 表A 这样的查询只会 ...
- splinter(python操作浏览器魔魁啊)
from splinter import Browser def main(): browser = Browser() browser.visit('http://google.com') brow ...
- 打造一个有感觉的vim(四)
今天要介绍的VIM插件的名字叫做surround,这个插件的主要作用是将一个VIM的tex-objects(文本对象)添加或者取消包裹(单引号,双引号,XML标签等等)下载地址如下:https://g ...