什么是 BDD

BDD(Behavior Driven Development),行为驱动开发,是一种敏捷软件开发的技术,它鼓励软件项目中的开发者、QA 和非技术人员或商业参与者之间的协作。

BDD 的重点是通过与利益相关者的讨论取得对预期的软件行为的清醒认识。它通过用自然语言书写非程序员可读的测试用例扩展了测试驱动开发方法。行为驱动开发人员使用混合了领域中统一的语言的母语语言来描述他们的代码的目的。这让开发者得以把精力集中在代码应该怎么写,而不是技术细节上,而且也最大程度的减少了将代码编写者的技术语言与商业客户、用户、利益相关者、项目管理者等的领域语言之间来回翻译的代价。

具体怎么操作

结合我们项目开发使用的 spring boot 2.x,下面我们来具体说明如何在实际项目中使用 BDD。

依赖的包

 1<cucumber.version>4.2.5</cucumber.version>
 2
 3...
 4
 5<dependency>
 6    <groupId>io.cucumber</groupId>
 7    <artifactId>cucumber-junit</artifactId>
 8    <version>${cucumber.version}</version>
 9    <scope>test</scope>
10</dependency>
11
12<dependency>
13    <groupId>io.cucumber</groupId>
14    <artifactId>cucumber-java</artifactId>
15    <version>${cucumber.version}</version>
16    <scope>test</scope>
17</dependency>
18
19<dependency>
20    <groupId>io.cucumber</groupId>
21    <artifactId>cucumber-spring</artifactId>
22    <version>${cucumber.version}</version>
23    <scope>test</scope>
24</dependency>

定义启动文件

BDD 其实也是依赖 junit,然后调用Cucumber的 Runner 来运行相应的测试。

 1package com.weibo.ad.bp.st.ryujo.web.test;
 2
 3import cucumber.api.CucumberOptions;
 4import cucumber.api.junit.Cucumber;
 5import org.junit.runner.RunWith;
 6
 7@RunWith(Cucumber.class)
 8@CucumberOptions(features = "classpath:features",
 9        tags = {"not @ignored", "@base"},
10        plugin = {"pretty", "html:target/cucumber", "junit:target/junit-report.xml"},
11        glue = {"classpath:com.weibo.ad.bp.st.ryujo.web.test.step"})
12public class RunCucumberTest {
13}
  • @CucumberOptions中的 features,用于指定我们项目中要运行的 feature 的目录
  • @CucumberOptions中的 format,用于指定我们项目中要运行时生成的报告,并指定之后可以在 target 目录中找到对应的测试报告
  • @CucumberOptions中的 glue,用于指定项目运行时查找实现 step 定义文件的目录
  • @CucumberOptions中的 tags,用来决定想要 Cucumber 执行哪个特定标签(以及场景),标签以“@”开头,如果是排除某个特定标签,用"not @ignored"

定义 feature

在项目模块的test/resources/features目录下新建一个get_mid_info.feature 文件

 1@base
 2Feature: Get Mid Info.
 3  This is some operations about mid.
 4
 5  Scenario Outline: Get Mid info by mid.
 6    Given mid is "<mid>"
 7    When I ask whether the mid info can be get correctly.
 8    Then I shoud be told "<answer>"
 9
10    Examples:
11      | mid        | answer |
12      | 2608812381 | Yes    |
13      | 123        | No     |

当然也可以使用Scenario来写

 1@base
 2Feature: Get Mid Info
 3    Scenario: I can get mid info correctly
 4      Given mid is 2608812381
 5      When I ask whether the mid info can be get correctly
 6      Then I shoud be told "Yes"
 7
 8    Scenario: I can't get mid info correctly
 9      Given mid is 123
10      When I ask whether the mid info can be get correctly
11      Then I shoud be told "No"

相关术语

单词中文含义
Feature功能
Background背景
Scenario场景,剧本
Scenario Outline场景大纲,剧本大纲
Examples例子
Given*, 假如,假设,假定
When*, 当
Then*, 那么
And*, 并且,而且,同时
But*, 但是

以上get_mid_info.feature文件中我们可以很清楚的了解到,我们这里定义了一个获取 mid info 的功能,此功能包含了根据 mid 获取 mid info 的场景大纲,大纲包含了示例的列表,假定 mid 依次为 examples 中列举的 mid 的值时候,当我们判断是否能正确获取到 mid info,那么答案依次为 examples 中对应的 answer。

实现相应的 step

我们在com.weibo.ad.bp.st.ryujo.web.test.step包下新建一个TestFeedMidServiceStep类:

 1@Slf4j
 2public class TestFeedMidServiceStep {
 3
 4    private Long mid;
 5
 6    private String actualAnswer;
 7
 8    @Autowired
 9    private FeedMidService feedMidService;
10
11    @Given("^mid is \"([^\"]*)\"$")
12    public void mid_is(String arg1) throws Exception {
13        this.mid = Long.valueOf(arg1);
14    }
15
16    @When("^I ask whether the mid info can be get correctly\\.$")
17    public void i_ask_whether_the_mid_info_can_be_get_correctly() throws Exception {
18        List<String> type = new ArrayList<>();
19        type.add("1");
20        FeedMidResp.ObjectEntity objectEntity =
21                feedMidService.getAuthorizedMidInfo(mid, null, "hosho", null, false, type, null, null, null);
22        log.info("{}", objectEntity);
23        if (null != objectEntity && !objectEntity.getItems().isEmpty()) {
24            this.actualAnswer = "Yes";
25        } else {
26            this.actualAnswer = "No";
27        }
28    }
29
30    @Then("^I shoud be told \"([^\"]*)\"$")
31    public void i_shoud_be_told(String arg1) throws Exception {
32        assertEquals(arg1, this.actualAnswer);
33    }
34}

配置 spring boot 容器

至此我们已经写好了一个基本的 feature,但是由于我们的测试依赖了 spring 管理的 bean,所以运行测试时必须启动 spring 容器。这里提供两种基本的方法。

方法一:

在每个 step 类上添加 spring 测试相关的注解:

1@Slf4j
2@ContextConfiguration
3@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
4@ActiveProfiles("test")
5public class TestFeedMidServiceStep {
6   ......
7}

或者自定义一个注解类CucumberStepsDefinition

 1package com.weibo.ad.bp.st.ryujo.web.test.step;
 2
 3import org.springframework.boot.test.context.SpringBootTest;
 4import org.springframework.test.context.ActiveProfiles;
 5import org.springframework.test.context.ContextConfiguration;
 6
 7import java.lang.annotation.ElementType;
 8import java.lang.annotation.Retention;
 9import java.lang.annotation.RetentionPolicy;
10import java.lang.annotation.Target;
11
12@Target(ElementType.TYPE)
13@Retention(RetentionPolicy.RUNTIME)
14@ContextConfiguration
15@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
16@ActiveProfiles("test")
17public @interface CucumberStepsDefinition {
18}

然后在每个 step 类上加上该注解:

1@Slf4j
2@CucumberStepsDefinition
3public class TestFeedMidServiceStep {
4   ......
5}

这种方法的优点是简单,基本上很多网上的示例代码都是这样写的,但是如果有多个 step 类的时候,在运行测试的时候,会多次初始化 spring 容器。而且还会抛出 WARN 信息。

于是就有了第二种方法。

方法二:

参考 https://github.com/cucumber/cucumber-jvm/issues/1420#issuecomment-405258386

glue指定的包路径下新建一个CucumberContextConfiguration

 1package com.weibo.ad.bp.st.ryujo.web.test.step;
 2
 3import cucumber.api.java.Before;
 4import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc;
 5import org.springframework.boot.test.context.SpringBootTest;
 6import org.springframework.test.context.ActiveProfiles;
 7
 8@ActiveProfiles("test")
 9@SpringBootTest
10@AutoConfigureMockMvc
11public class CucumberContextConfiguration {
12
13    @Before
14    public void setup_cucumber_spring_context(){
15        // Dummy method so cucumber will recognize this class as glue
16        // and use its context configuration.
17    }
18}

注意这里用的@Before注解是cucumber.api.java.Before,如果用错了,是不起作用的。

然后我们就可以正常的实现 step 类了,而且可以直接使用@Autowired来注入 spring 管理的 bean.

运行测试

1mvn clean test

或者使用 IDE 的话,直接点击RunCucumberTest类上的按钮即可。

补充说明

IDEA 建议安装 Cucumber for java 插件。https://plugins.jetbrains.com/plugin/7212-cucumber-for-java

中文支持

Cucumber 本身支持超过 30 种语言(此处语言是 Spoken Language 而非 Programming Language)。查看其所有支持的语言和对应的关键字可以访问https://github.com/cucumber/cucumber/blob/master/gherkin/gherkin-languages.json

下面我们来使用中文来描述一个 feature。

 1# language: zh-CN
 2@base
 3功能: 测试计算器
 4
 5  场景大纲: 两个整数的四则运算
 6    假如 我们有两个整数<a>和<b>
 7    当 我们将其求和运算
 8    那么 我们能得到数值 <c>
 9    例子:
10
11      | a | b | c |
12      | 1 | 2 | 3 |
13      | 2 | 5 | 7 |

首先第一行我们使用# language: zh-CN来说明我们使用的是中文,这样安装了 cucumber 插件的 IDE 也会有相应的关键字提示和语法高亮。

然后编写 step

 1package cn.iliubang.exercises.bdd.test.glue;
 2
 3import cucumber.api.java.zh_cn.假如;
 4import cucumber.api.java.zh_cn.当;
 5import cucumber.api.java.zh_cn.那么;
 6import org.junit.Assert;
 7
 8public class Test1 {
 9
10    private Integer a;
11    private Integer b;
12    private Integer c;
13
14    @假如("我们有两个整数{int}和{int}")
15    public void 我们有两个整数_和(Integer int1, Integer int2) {
16        a = int1;
17        b = int2;
18    }
19
20    @当("我们将其求和运算")
21    public void 我们将其求和运算() {
22        c = a + b;
23    }
24
25    @那么("我们能得到数值 {int}")
26    public void 我们能得到数值(Integer int1) {
27        Assert.assertEquals(int1, c);
28    }
29}

这里很神奇的是,连注解都是中文的。最后是编写启动文件,这里不再赘述。运行的结果如下:

指定 Cucumber 运行结果报告

Cucumber 本身支持多种报告格式以适用于不同环境下调用的报告输出:

  • pretty :用于在命令行环境下执行 Cucumber 测试用例所产生的报告,如果您的 console 支持,pretty 形式的报告还可以按照颜色显示不同的运行结果;如下图所示的例子分别显示了用例执行通过和用例没有 Steps definitions 的输出报告:

  • json :多用于在持续集成环境下的跨机器生成报告时使用,比如在用例执行的机器 A 上运行 Cucumber 测试用例,而在调度或报告机器 B 上生成用例执行报告,此时只需要把生成的 JSON 报告传输到机器 B 上即可。
  1[
  2  {
  3    "line": 3,
  4    "elements": [
  5      {
  6        "line": 12,
  7        "name": "两个整数的四则运算",
  8        "description": "",
  9        "id": "测试计算器;两个整数的四则运算;;2",
 10        "type": "scenario",
 11        "keyword": "场景大纲",
 12        "steps": [
 13          {
 14            "result": {
 15              "duration": 3204397,
 16              "status": "passed"
 17            },
 18            "line": 6,
 19            "name": "我们有两个整数1和2",
 20            "match": {
 21              "arguments": [
 22                {
 23                  "val": "1",
 24                  "offset": 7
 25                },
 26                {
 27                  "val": "2",
 28                  "offset": 9
 29                }
 30              ],
 31              "location": "Test1.我们有两个整数_和(Integer,Integer)"
 32            },
 33            "keyword": "假如"
 34          },
 35          {
 36            "result": {
 37              "duration": 110745,
 38              "status": "passed"
 39            },
 40            "line": 7,
 41            "name": "我们将其求和运算",
 42            "match": {
 43              "location": "Test1.我们将其求和运算()"
 44            },
 45            "keyword": "当"
 46          },
 47          {
 48            "result": {
 49              "duration": 1611672,
 50              "status": "passed"
 51            },
 52            "line": 8,
 53            "name": "我们能得到数值 3",
 54            "match": {
 55              "arguments": [
 56                {
 57                  "val": "3",
 58                  "offset": 8
 59                }
 60              ],
 61              "location": "Test1.我们能得到数值(Integer)"
 62            },
 63            "keyword": "那么"
 64          }
 65        ],
 66        "tags": [
 67          {
 68            "name": "@base"
 69          }
 70        ]
 71      },
 72      {
 73        "line": 13,
 74        "name": "两个整数的四则运算",
 75        "description": "",
 76        "id": "测试计算器;两个整数的四则运算;;3",
 77        "type": "scenario",
 78        "keyword": "场景大纲",
 79        "steps": [
 80          {
 81            "result": {
 82              "duration": 779875,
 83              "status": "passed"
 84            },
 85            "line": 6,
 86            "name": "我们有两个整数2和5",
 87            "match": {
 88              "arguments": [
 89                {
 90                  "val": "2",
 91                  "offset": 7
 92                },
 93                {
 94                  "val": "5",
 95                  "offset": 9
 96                }
 97              ],
 98              "location": "Test1.我们有两个整数_和(Integer,Integer)"
 99            },
100            "keyword": "假如"
101          },
102          {
103            "result": {
104              "duration": 472993,
105              "status": "passed"
106            },
107            "line": 7,
108            "name": "我们将其求和运算",
109            "match": {
110              "location": "Test1.我们将其求和运算()"
111            },
112            "keyword": "当"
113          },
114          {
115            "result": {
116              "duration": 199326,
117              "status": "passed"
118            },
119            "line": 8,
120            "name": "我们能得到数值 7",
121            "match": {
122              "arguments": [
123                {
124                  "val": "7",
125                  "offset": 8
126                }
127              ],
128              "location": "Test1.我们能得到数值(Integer)"
129            },
130            "keyword": "那么"
131          }
132        ],
133        "tags": [
134          {
135            "name": "@base"
136          }
137        ]
138      }
139    ],
140    "name": "测试计算器",
141    "description": "",
142    "id": "测试计算器",
143    "keyword": "功能",
144    "uri": "classpath:features/test_1.feature",
145    "tags": [
146      {
147        "name": "@base",
148        "type": "Tag",
149        "location": {
150          "line": 2,
151          "column": 1
152        }
153      }
154    ]
155  }
156]
  • html :用于生成简单的 HTML 格式的报告以便查看 Cucumber 测试用例运行的结果

  • junit :用于生成 JUnit 格式的报告:
 1<?xml version="1.0" encoding="UTF-8" standalone="no"?>
 2<testsuite failures="0" name="cucumber.runtime.formatter.JUnitFormatter" skipped="0" tests="2" time="0.028641">
 3<testcase classname="测试计算器" name="两个整数的四则运算" time="0.022812">
 4<system-out><![CDATA[假如我们有两个整数1和2................................................................passed
 5当我们将其求和运算...................................................................passed
 6那么我们能得到数值 3.................................................................passed
 7]]></system-out>
 8</testcase>
 9<testcase classname="测试计算器" name="两个整数的四则运算_2" time="0.005829">
10<system-out><![CDATA[假如我们有两个整数2和5................................................................passed
11当我们将其求和运算...................................................................passed
12那么我们能得到数值 7.................................................................passed
13]]></system-out>
14</testcase>
15</testsuite>

此外,Github 上有很多开源的插件或者 Cucumber 扩展可以帮助从 JSON 格式的报告生成 HTML 格式的报告。这里推荐大家使用 Cucumber-reporting。Cucumber-reporting 不仅能够完成从 JSON 格式报告生成 HTML 格式报告,而且可以按照 tag 和 feature 以及 step 查看,不得不提的是生成的 HTML 格式报告的样式非常好看,下面就是以本文中所使用的 feature 文件为例,以 Cucumber-reporting 来生成的 HTML 报告: