代码改变世界

python BDD 框架之lettuce

2013-11-09 16:22  虫师  阅读(17514)  评论(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()方法被调用时的处理过程,对参数的内容转换成整数,判断如果等于01的话就直接返回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 最通俗易懂的文章了,没有之一,给个赞吧!

 

Web Page Counters
Computer Desks