Cucumber 黄瓜测试 BDD 从入门到精通

1. Cucumber

Cucumber 是 BDD(Behavior-Driven Development,行为驱动开发)的一个自动化测试工具,使用自然语言来描述测试用例,使得 非研发(QA、PM)也可以理解甚至编写 测试用例。

官方表示:应该将 Cucumber 视为一个【文档编写工具】,而非一个单纯的自动化测试工具

撰写时,应该要以 PM 也能理解 测试用例 为目标去编写 Cucumber

2. Gherkin

Gherkin 是 Cucumber 用来描述 测试用例 的语言,以下为关键字的用意与关联关系。

以 分享概览 为例

Given:新增 帐号abc@qq.com 、新增 概览1

When:将 概览1 分享给 帐号abc@qq.com

Then:校验 帐号abc@qq.com 是否能查看到 概览1

2.1 Scenario 范例

Feature: 授权功能Scenario: 帐号 通过绑定 角色 进行授权        Given 新增角色'分析师',拥有权限'1001, 1002, 1003'And 新增帐号'abc@qq.com'When 帐号'abc@qq.com'绑定角色'分析师'Then 鉴权 帐号'abc@qq.com',有权限'1001'And  鉴权 帐号'abc@qq.com',有权限'1002'But  鉴权 帐号'abc@qq.com',没有权限'1005'复制代码

2.2 Background 范例

Feature: 授权功能Background:      Given 新增帐号'abc@qq.com'Scenario: 帐号 通过绑定 角色 进行授权        Given 新增角色'分析师',拥有权限'1001, 1002, 1003'When 帐号'abc@qq.com'绑定角色'分析师'Then 鉴权 帐号'abc@qq.com',有权限'1001'And  鉴权 帐号'abc@qq.com',有权限'1002'But  鉴权 帐号'abc@qq.com',没有权限'1005'Scenario: 帐号 通过绑定 机构 进行授权        Given 新增机构'北京部门',拥有权限'2001, 2002, 2003'When 帐号'abc@qq.com'绑定机构'北京部门'Then 鉴权 帐号'abc@qq.com',有权限'2001'And  鉴权 帐号'abc@qq.com',有权限'2002'But  鉴权 帐号'abc@qq.com',没有权限'2005'复制代码

2.3 Scenario Outline 范例

可以在多个 Step 上共用同一个 "简单" 参数,且每一个 Example 都视为一个 Scenario

Feature:授权功能Scenario Outline:帐号通过绑定角色进行授权Given新增角色<role>,拥有权限<permissions>When新增帐号<account>Then帐号<account>绑定角色<role>And鉴权帐号<account>,有权限<has_permission>Examples:|role|permissions|account|has_permission||分析师|1001,1002,1003|abc@qq.com|1001||开发者|2001,2002,2003|cde@qq.com|2001||管理员|3001,3002,3003|fgh@qq.com|3001|复制代码

3. 基本概念

3.1 文件结构

Gherkin 写在 .feature 文件中

Step 对应的逻辑 写在 .java 文件中

3.2 Step 映射

通过 Gherkin 语法上的描述,找到与 注解 value 值匹配的 Java 方法,将 Gherkin 与 Java 代码关联起来。

3.3 Scenario 独立

当同时执行多个 Scenario 时,执行每个 Scenario 对应的 Java 文件都会被重新创建。

不同的 Scenario 之间,不应该存在数据依赖(MySQL),如果存在依赖,将会使 Scenario 变得脆弱可以在 Backgroud,进行数据清理,来保证测试结果的正确性

二、最佳实践

1. 撰写 Scenairo 原则 - BRIEF

school.cucumber.io/courses/tak…

B:Business Language。

Scenairo 中使用的词语应该使用【业务团队成员】能够理解的词语,否则将无法与业务团队成员互动。

R:Real Data。

Scenairo 中应该使用 具体、真实 的数据(不要用 1、2、3、A、B、C),有助于让场景变得生动,并及早揭示边界条件与基本假设。

I:Intention Revealing。

Scenairo 应该描述试图实现的意图,而不是描述程式将如何实现它的机制。

确保每一行 Step 描述的是 意图 而非 机制。 (比如:创建帐号,就不要写成 "将帐号数据写入 user 表,并在 account_project 表绑定帐号与项目的关联")

E:Essential。

Scenairo 应该只保留必要的 Step,不直接促成结果的场景都应该被删除。

任何不能增加读者对预期行为理解的场景,都不應該出现在文档中。

F:Focus。

多数的 Scenairo 应该只专注于单一职责。

BRIEF

建议将大多数的 Scenairo 限制在五行或更少,这将使它们更易于阅读与推理,并有助于避免 同时测试多个规则 或 增加额外细节。

2. 保证 Scenairo 可读性好处

school.cucumber.io/courses/tak…

随时获得 你做的事情是否正确 的反馈

你的 Feature 可以变成描述你 系统功能 的 线上文档

Scenairo 将会引导你的技术设计

3. 开发流程推荐

school.cucumber.io/courses/tak…

在 Cucumber 中描述你想要实现的 Scenairo,把所有的 Step 串连起来,并运行 Cucumber 使其出现 失败 结果

持续实现 Step 与 API 的具体逻辑,并观察 API 是如何 失败 的,最终使 Scenairo 的结果变为 成功。

当测试通过后,对 API 实现进行 清理 与 优化(重构),使其更具可读性,并再次运行 Cucumber 保证重构后的结果正确。

以上为 测试驱动开发(TDD) 的 生命周期: Red、Green、Clean

如果改坏了系统逻辑,你的测试用例会告诉你。

推荐:在研发进行技术设计前,再多加一个 测试用例评审 的环节,让 PM、QA、RD 一起参与,方便及早发现问题,也能增加技术设计时考量的全面性

三、Cucumber 常用功能

1. 参数化

参数化 可以与 表格化、列表化、对象化 混用

1.1 关键字

类型正则

biginteger"-?\d+" 或者 "\d+"

string"([^"\] (\.[^"\] )*)"

bigdecimal"-?\d*[.,]\d+"

byte"-?\d+" 或者 "\d+"

double"-?\d*[.,]\d+"

short"-?\d+" 或者 "\d+"

float"-?\d*[.,]\d+"

word"\w+"

int"-?\d+" 或者 "\d+"

long"-?\d+" 或者 "\d+"

1.2 说明

Cucumber 支持在 Java 注解 中使用 {关键字} 作为占位符。 在 Step 中直接写上参数,将在 Java 代码中,会把占位符对应的参数作为方法参数传递进去。

字串 类型的 关键字,需要加上 单引号 或 双引号 作为声明

注解中声明占位符的顺序 为 注入方法参数的顺序

1.3 范例

Feature: 授权功能Scenario: 鉴权场景          Given 创建帐号'waiting001@qq.com',角色'admin',项目id1复制代码

@Given("创建帐号 {string},角色 {string},项目id {int}")publicvoidtest(Stringusername,Stringrole, Integer projectId) {      log.info("start to execute test(), params:[ username = {}, role = {}, projectId = {} ]",              username, role, projectId);  }复制代码

2. 表格化(DataTable)

2.1 设置 Gherkin 数据

下方的 List - Map、List - List、Map - List 都是共用同一套 Gherkin 代码,也就是说,同一个 Gherkin 代码,Cucumber 可以根据不同的方法参数类型,自动进行转换

Feature: 授权功能Scenario: 鉴权场景        Given 创建帐号        | username          | password | role      | project_id |        | waiting001@qq.com | a123456  | admin    |1|        | waiting002@qq.com | b123456  | analyst  |2|        | waiting003@qq.com | c123456  | developer |3|复制代码

2.2 List - Map(常用)

@Given("创建帐号")publicvoid test(List> dataTable) {for(Mapdata: dataTable) {          String username =data.get("username");          String password =data.get("password");          String role =data.get("role");          String projectId =data.get("project_id");          log.info("execute test(), fields:[ username = {}, password = {}, role = {}, projectId = {} ]",                  username, password, role, projectId);      }    }复制代码

2.3 List - List

@Given("创建帐号")publicvoid test(List> dataTable) {for(Listdata: dataTable) {          log.info("execute test(), fields:[ data = {} ]",data);      }    }复制代码

2.4 Map - List

@Given("创建帐号")  publicvoidtest(Map> dataTable) {for(Map.Entry> data : dataTable.entrySet()) {Stringusername = data.getKey();          List infos = data.getValue();          log.info("execute test(), fields:[ username = {}, infos = {} ]", username, infos);      }    }复制代码

3. 列表化

3.1 直列表

Feature: 授权功能Scenario: 鉴权场景        Given 删除帐号        | waiting001@qq.com |        | waiting002@qq.com |        | waiting003@qq.com |复制代码

@Given("删除帐号")publicvoidtest(List<String> usernames){log.info("execute test(), fields:[ usernames = {} ]", usernames);  }复制代码

3.2 横列表

Feature: 授权功能Scenario: 鉴权场景        Given 删除帐号        | waiting001@qq.com | waiting002@qq.com | waiting003@qq.com |复制代码

@Given("删除帐号")  public void test(@TransposeList usernames) {log.info("execute test(), fields:[ usernames = {} ]", usernames);  }复制代码

记得加上 @Transpose,告诉 Cucumber 需要进行数据转换

4. 对象化

4.1 撰写 Gherkin 语法

Feature: 授权功能Scenario: 鉴权场景        Given 新增 帐号        | username        | role      | project    |        | waiting1@qq.com | admin    | default    |        | waiting2@qq.com | analyst  | production |        | waiting3@qq.com | developer | default    |复制代码

4.2 定义 Java 对象

@DatastaticclassAccount {/** 

    * 帐号 

    */privateStringusername;/** 

    * 角色 

    */privateStringrole;/** 

    * 项目 

    */privateStringproject;    }复制代码

4.3 撰写 封装 Java 对象的 方法

/** 

* 帐号对象 的 封装方法 

*/@DataTableTypepublicAccount defineAccount(Map entry) {        Account account = new Account();// 如果有指定数据,则将数据写入  Optional.ofNullable(entry.get("username")).ifPresent(account::setUsername);      Optional.ofNullable(entry.get("role")).ifPresent(account::setRole);  Optional.ofNullable(entry.get("project")).ifPresent(account::setProject);returnaccount;  }复制代码

在封装方法上方,需要加上 @DataTableType 注解

4.4 关联具体 Step 方法

@Given("新增 帐号")publicvoidaddAccount(List<Account> accountList){for(Account account : accountList) {log.info("execute addAccount(), fields:[ account = {} ]", account);      }    }复制代码

方法参数中,直接指定 对象封装方法 返回的 对象类型,Cucumber 就能直接进行关联

5. 参数化、表格化、列表化 混合使用

DataTable 与 List 必须作为 Java 方法的最后一个参数

5.1 表格化&参数化

Feature: 授权功能Scenario: 鉴权场景        Given 创建帐号,项目'default'| username          | password | role      |        | waiting001@qq.com | a123456  | admin    |        | waiting002@qq.com | b123456  | analyst  |        | waiting003@qq.com | c123456  | developer |复制代码

@Given("创建帐号,项目 {string}")  publicvoidtest(Stringproject, List> dataTable) {        log.info("execute test(), fields:[ project = {} ]", project);for(Map data : dataTable) {Stringusername = data.get("username");Stringpassword = data.get("password");Stringrole = data.get("role");          log.info("execute test(), fields:[ username = {}, password = {}, role = {} ]",                  username, password, role);      }    }复制代码

List<Map<String, String>> 必须作为最后一个方法参数

5.2 直列表&参数化

Feature: 授权功能Scenario: 鉴权场景        Given 删除帐号,项目'default'| waiting001@qq.com |        | waiting002@qq.com |        | waiting003@qq.com |复制代码

@Given("删除帐号,项目 {string}")publicvoidtest(StringprojectName, List usernames) {      log.info("execute test(), fields:[ projectName = {}, usernames = {} ]", projectName, usernames);  }复制代码

List 必须作为最后一个方法参数

5.3 横列表&参数化

Feature: 授权功能Scenario: 鉴权场景        Given 删除帐号,项目'default'| waiting001@qq.com | waiting002@qq.com | waiting003@qq.com |复制代码

@Given("删除帐号,项目 {string}")  public void test(String project,@TransposeList usernames) {log.info("execute test(), fields:[ project = {}, usernames = {} ]", project, usernames);  }复制代码

5.4 对象化&参数化

Feature: 授权功能Scenario: 鉴权场景        Given 新增 帐号,机构'北京部门'| username        | role      | project    |        | waiting1@qq.com | admin    | default    |        | waiting2@qq.com | analyst  | production |        | waiting3@qq.com | developer | default    |复制代码

@Given("新增 帐号,机构 {string}")publicvoidaddAccount(String organization, List<Account> accountList){log.info("execute addAccount(), fields:[ organization = {} ]", organization);for(Account account : accountList) {log.info("execute addAccount(), fields:[ account = {} ]", account);      }    }复制代码

6. 钩子方法(Hook)

6.1 注解

注解执行时机

@BeforeAll在启动 Cucumber 时执行

@Before在所有 Scenario 执行之前执行

@BeforeStep在所有 Step 执行之前执行

@AfterAll在结束 Cucumber 时执行

@After在所有 Scenario 执行之后执行

@AfterStep在所有 Step 执行之后执行

指定注解的 value,可以指定 Hook 的运行范围,不指定则表示在全项目生效。 注解可以定义在项目中的任意位置。

如果同时定义了多个钩子方法,则会依照注解中 order 属性的顺序多次执行。

6.2 全局生效范例

Gherkin 代码

Feature:测试功能  Scenario:测试场景1  Given测试功能1  When测试功能2  Then测试功能3  Scenario:测试场景2  Given测试功能1  When测试功能2  Then测试功能3复制代码

Java 代码

@Slf4jpublicclassTestDefs{@Beforepublicvoidbefore(){          log.info("执行 @Before 方法");      }@BeforeSteppublicvoidbeforeStep(){          log.info("执行 @BeforeStep 方法");      }@Afterpublicvoidafter(){          log.info("执行 @After 方法");      }@AfterSteppublicvoidafterStep(){          log.info("执行 @AfterStep 方法");      }@Given("测试功能1")publicvoidtest1(){          log.info("执行 测试功能1");      }@Given("测试功能2")publicvoidtest2(){          log.info("执行 测试功能2");      }@Given("测试功能3")publicvoidtest3(){          log.info("执行 测试功能3");      }  }复制代码

最终结果

6.3 局部生效范例

Gherkin 代码在不同的 Scenario 上,可以增加 自定义注解 作为标记

Feature: 测试功能@MyHookScenario: 测试场景1Given 测试功能1When  测试功能2@YourHookScenario: 测试场景2Given 测试功能1When  测试功能2复制代码

Java 代码Hook 注解的 value 值,可以用来指定,只有当 Scenario 上有该自定义注解时,才会执行 Hook 方法。

如果想将 Hook 方法在多个自定义注解下进行复用,可以通过 , 隔开,例如:@Before("@MyHook,@YourHook")

@Slf4jpublicclassTestDefs{@Before("@MyHook")publicvoidmyHookBefore(){          log.info("执行 @MyHook 的 @Before 方法");      }@BeforeStep("@MyHook")publicvoidmyHookBeforeStep(){          log.info("执行 @MyHook 的 @BeforeStep 方法");      }@Before("@YourHook")publicvoidyourHookBefore(){          log.info("执行 @YourHook 的 @Before 方法");      }@BeforeStep("@YourHook")publicvoidyourHookBeforeStep(){          log.info("执行 @YourHook 的 @BeforeStep 方法");      }@Given("测试功能1")publicvoidtest1(){          log.info("执行 测试功能1");      }@When("测试功能2")publicvoidtest2(){          log.info("执行 测试功能2");      }    }复制代码

最终结果

四、Gherkin 模版范例

0. Gherkin 模版撰写规范

所有 模版功能 只运行在 测试项目(project_id = 1) (避免有时要指定 project_id 有时又不需要的麻烦)

使用 Gherkin 参数化 的 字串 功能时,使用 单引号 将字串包裹 (考虑到模版参数可能出现 Json 字串,使用双引号会需要额外进行转译)

模版参数中,不要出现 id(包含 帐号id、角色id、职务id ... 等),应该改用 名称(用户名、角色中文名称、职务名称)去反查 (使用 id 将无可避免的出现要使用 上下文 将多个 Step 进行关联的问题,这将使测试用例变得脆弱,所以宁可用 名称 去反查 id,也不要直接指定 id)

Gherkin 语法表格化的表头字段,使用 下滑线命名法 来命名

模版方法名称定义遵守以下规范 (当一个模版可能同时存在于 Given 与 When 时,以 When 为主)

模版关键字说明范例

Given以 [模块][新增or更新or删除] + 空格 开头[帐号][新增] 新增帐号、[角色][更新] 更新角色

When以 [模块][功能] + 空格 开头[权限][功能] 授权数据资源权限、[权限][功能] 删除数据资源权限数据

Then以 [模块][校验] + 空格 开头[权限][校验] 校验帐号权限

随时注意编写的模版是否尽量符合第 2 大项的最佳实践

1. 帐号

通常为 Given 的最后一步,比如 帐号 要绑定 角色,一定是先创建好 角色,再通过 新增帐号 进行绑定

1.1 新增 帐号

(创建前自动删除原来的帐号、创建时允许指定角色)

属性说明必填默认值

username用户名是无

password密码否AAAaaa111@

user_cname帐号中文名称否与 username 相同

email信箱否与 username 相同

role_cname角色中文名称,默认角色用 role_name 即可否admin

# 最简单的范例Given [帐号][新增] 新增帐号    | username          |    | waiting001@qq.com |    | waiting002@qq.com |    | waiting003@qq.com |# 最完整的范例Given [帐号][新增] 新增帐号    | username          | password  | user_cname        | email            | role_cname |    | waiting001@qq.com | AAAaaa111@ | waiting001@qq.com | waiting001@qq.com | admin      |    | waiting002@qq.com | AAAaaa111@ | waiting002@qq.com | waiting002@qq.com | analyst    |    | waiting003@qq.com | AAAaaa111@ | waiting003@qq.com | waiting003@qq.com | developer  |复制代码

1.2 更新 帐号 的 角色

Given [帐号][更新] 更新帐号 'waiting001@qq.com' 的角色    | 角色A | 角色B | 角色C |复制代码

1.3 更新 帐号 的 职务

Given [帐号][更新] 更新帐号 'waiting001@qq.com' 的职务    | 职务A | 职务B | 职务C |复制代码

1.4 删除 帐号

Given [帐号][删除] 删除帐号    | waiting001@qq.com |    | waiting002@qq.com |    | waiting003@qq.com |    | waiting004@qq.com |复制代码

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,133评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,682评论 3 390
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,784评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,508评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,603评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,607评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,604评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,359评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,805评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,121评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,280评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,959评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,588评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,206评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,193评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,144评论 2 352

推荐阅读更多精彩内容