python BDD 框架之lettuce
2013-11-09 16:22 虫师 阅读(17540) 评论(7) 编辑 收藏 举报
http://pythonhosted.org/lettuce/
介绍
亲们!看到TDD/BDD 一定会感觉高端大气上档次,不是我们普通吊民玩的,最的近在一直在摸索自动化测试。也想体验并引入BDD 低调奢华的内涵。于是,在网络上搜索资料;话说这玩艺儿真的不太好理解,尤其对于没有丰富编程的经验的同学。
学习BDD ruby 的cucumber 是个不错的选择,但我是python 流的,自然找了来它的兄弟lettuce ,从官方版本(v0.1rc11)来看确实够年轻的,不过由ruby 的cucumber 在前面开路,lettuce 应该会发展的很顺利。
lettuce 除了官方文档外,几乎找不到其它资料,为了理解lettuce ,我们不妨多去看看cucumber 的资料。
lettuce是一个非常有用的和迷人的BDD(行为驱动开发)工具。Python项目的自动化测试,它可以执行纯文本的功能描述,就像ruby语言的cucumber。
lettuce,使开发和测试过程变得很容易,可扩展性,可读性和-什么是最好的-它允许我们用自然语言去描述个一个系统的行为,你不能想象这些描述可以自动测试你的系统。
安装
请确认你已经安装了python 以及pip安装包管理工具。
不管是windows 还是linux 环境,进入pip目录,只需下面一个命令就可以安装lettuce .
user@machine:~$ [sudo] pip install lettuce
例子(阶乘)
下面就通过官网的例子来领略lettuce的风骚。
什么阶乘?
0!=1
1!=1
2!=2×1=2
3!=3×2×1=6
....
10!=10×9×8×7×6×5×4×3×2×1=3628800
.....
下面是用python 语言的两种阶乘实现:
#coding=utf-8 #循环实现 def f(n): c = 1 for i in range(n): i=i+1 c=c*i return c #递归实现 def f2(n): if n > 1: return n*f2(n-1) else: return 1 #调用方法 print f(10) print f2(10)
OK!介于我们理解上面阶乘的基础上,来看看BDD是如何实现的。
lettuce 解析
创建以下目录结构:
.../tests/features/zero.feature
/steps.py
现在我们来编写zero.feature 文件的内容
Feature: Compute factorial
In order to play with Lettuce
As beginners
We'll implement factorial
Scenario: Factorial of 0
Given I have the number 0
When I compute its factorial
Then I see the number 1
基于E文不好的同学,我这里简单翻译一下:
功能:计算阶乘
为了使用lettuce
作为初学者
我们将实现阶乘
场景:0的阶乘
假定我有数字0
当我计算它的阶乘
然后,我看到了1
是不是很自然的描述?!第一段功能介绍,我要实现什么功能;第二段场景,也可以看做是一条测试用例,当我输入什么数据时,程序应该返回给我什么数据。
虽然是自然语言,但还是有语法规则的,不然一千个人有一千中描述,程序以什么样的规则去匹配呢?
其实它的语法规则非常简单就几个关键字,记住他们的含义及作用即可。
----------------------------------------------
- Feature(功能)
- Scenario(情景)
- Given(给定)
- And(和)
- When(当)
- Then(则)
----------------------------------------------
他们的含义与原有自动化测试框架的概念相似,类比如下:
关于feature文件的作用,执行以及语法规则将在下一节中详细介绍,这一节主要先来体验luttuce的风骚。
有了上面zero.feature文件的行为做指导,下面打开steps.py 文件来编写我们的程序。
from lettuce import * @step('I have the number (\d+)') def have_the_number(step, number): world.number = int(number) @step('I compute its factorial') def compute_its_fatorial(step): world.number = factorial(world.number) @step('I see the number (\d+)') def check_number(step, expected): expected = int(expected) assert world.number == expected, \ "Got %d" % world.number def factorial(number): number = int(number) if (number == 0) or (number == 1): return 1 else: return number
我擦!咋一看怎么跟我上面实现阶乘的代码相差甚远呀!不知道你和你的小伙伴有没有惊呆!?好吧,以我拙劣的python语言水平试着来分析一下,这是啥?这是啥?这又是啥?
from lettuce import *
引入lettuce 下面的所有包
@step('I have the number (\d+)')
@step 字面意思是步骤
I have the number (\d+) 对应的就是zero.feature文件中的第六句:Given I have the number 0
(\d+) 是一个正则表达式,\d 表示匹配一个数字,+ 表示匹配的数字至少有一个或多个。关于这个可以参考其他python 正则表达式的资料。
第一步:
@step('I have the number (\d+)') def have_the_number(step, number): world.number = int(number)
定义一个方法have_the_number,把假设的输入(0)转换成整型放入world.number中。
第二步:
@step('I compute its factorial') def compute_its_fatorial(step): world.number = factorial(world.number)
把have_the_number方法中world.number的变量值(0)放入 factorial() 方法中,并把结果返再赋值给world.number变量。
I compute its factorial 对应的就是zero.feature文件中的第七句:When I compute its factorial
第三步:
def factorial(number): number = int(number) if (number == 0) or (number == 1): return 1 else: return number
这个是factorial()方法被调用时的处理过程,对参数的内容转换成整数,判断如果等于0或1的话就直接返回1,否则返回具体的数。(处理结果给了第三步的world.number)
第四步:
@step('I see the number (\d+)') def check_number(step, expected): expected = int(expected) assert world.number == expected, \ "Got %d" % world.number
expected 获取的是zero.feature文件中预期结果, 与第三步处理的实际结果(world.number)进行比较;assert 函数进行断言结果是否正确。
I see the number (\d+)对应的就是zero.feature文件中的第八句:Then I see the number 1
运行
切换到tests目录下,运行lettuce 命令:
fnngj@fnngj-H24X:~/python/lettuce/tests$ lettuce
运行结果如下:
运行结果很清晰,首先是zero.feature文件里功能描述(feature),然后场景(scenario)每一步所对应steps.py 文件里的哪一行代码。
最后给出运行结果:
Feature(1 passed) 一个功能通过
Scenario(1 passed) 一个场景通过
Steps(3 passed) 三个步骤通过
完善
下面我们可以在zero.feature 中多加几个场景(测试用例):
Feature: Compute factorial
In order to play with Lettuce
As beginners
We'll implement factorial
Scenario: Factorial of 0
Given I have the number 0
When I compute its factorial
Then I see the number 1
Scenario: Factorial of 1
Given I have the number 1
When I compute its factorial
Then I see the number 1
Scenario: Factorial of 2
Given I have the number 2
When I compute its factorial
Then I see the number 2
Scenario: Factorial of 3
Given I have the number 3
When I compute its factorial
Then I see the number 6
运行结果:
嗯??第四场景没通过,3!=3*2*1=6 这个预期结果肯定是正确的,那就是代码的逻辑有问题吧!如果你细心的话一定发现了setup.py中的代码并未真正实现阶乘,我们需要对它进行修改:
#coding=utf-8 from lettuce import * @step('I have the number (\d+)') def have_the_number(step, number): world.number = int(number) print world.number @step('I compute its factorial') def compute_its_fatorial(step): world.number = factorial(world.number) print world.number @step('I see the number (\d+)') def check_number(step, expected): expected = int(expected) assert world.number == expected, \ "Got %d" % world.number def factorial(number): number = int(number) if (number == 0) or (number == 1): return 1 else: return number*factorial(number-1)
代码修改部分:
def factorial(number): number = int(number) if (number == 0) or (number == 1): return 1 else: return number*factorial(number-1)
参照本文开头,通过递归的方式实现阶乘的代码,现在才算完整的实现阶乘。OK !再来运行以下脚本吧!全绿了有木有!!
这应该是目前讲解BDD 最通俗易懂的文章了,没有之一,给个赞吧!