前段时间读到一篇优秀的文章《前端开源项目持续集成三剑客》,就想试着运用到自己的项目中去。(好吧,老实说,我只是个徽章收集爱好者。)
持续集成,这个概念对后端来说应该并不陌生,甚至可以说是司空见惯吧。但是,这对曾经(除了那些大厂)单元测试都不一定要写的前端来说,或许是个陌生的词。
然而,随着前端飞速地发展,不断吸取后端长久以来积累的经验,以及前端对单元测试越来越重视,持续集成作为前端工程化中的一项也渐渐进入人们的视野。
那么,持续集成究竟是什么?
持续集成(英语:Continuous integration,缩写为 CI),一种软件工程流程,将所有工程师对于软件的工作复本,每天集成数次到共用主线(mainline)上。 —— wikipedia
简单来说,就是以一定的频率将代码整合到一起。
使用持续集成能使项目:
保持可测试和可发布的状态
易于追踪错误,当集成产生错误时,能将错误产生的缩小范围到上次成功集成之后的提交
版本回滚也变得轻而易举
在《前端开源项目持续集成三剑客》中,作者推荐了 2 个集成工具,分别是:travis-ci 和 circleci。
额...该选哪个哪?
分别粗略地了解了这两个产品,它俩的网站的都非常简洁,文档也很清晰,功能上也大致相同。虽然,circleci 比 travis-ci 多了 Bitbucket 源码库的支持,但是,有一大硬伤 circleci 只对一个 container 免费,而且,若使用 OS X 需要额外收费。与之相反,travis-ci 只要是 Github 上的开源项目全部免费,且支持在 OS X 运行。
Travis-ci。
注册 travis 只需一步,点击 Sign In 按钮绑定 Github。登录后,执行 travis 只需以下 3 步:
添加需要 travis 管理的项目
为项目添加 .travis.yml 配置文件
提交代码
与此同时,travis 的配置也极其简单。如果没有什么特别的需求,那么,只需配置运行语言类型及其版本就行。
// .travis.yml
language: node_js
node_js:
- "6"
这样,一个简单、可用的 travis 配置就完成了。
Travis 构建过程主要分为两步:
install:安装依赖,在 node 环境下,默认运行 npm install
stript:运行构建命令,在 node 环境下,默认运行 npm test
那么,上面的代码就等价于:
language: node_js
node_js:
- "6"
install: npm install
script: npm test
当然,travis 不止这两个生命周期,额外的配置需求都可以到官网查看。
OK。提交代码试试吧。
travis 的运行信息都可以在 Job log 中看到。
如果运行成功,你就可以通过 https://img.shields.io/travis... 或 https://img.shields.io/travis... 来给你的项目添加 badge 了,就像这样 。
Tips:其中的 USER, REPO, BRANCH 都要替换成个人信息。
有了构建的徽章,接着再弄一个测试覆盖率的徽章。三剑客文章中用的是 coveralls,但进入它的官网发现,它和当今网站那种简洁风格不同,画风有点 classic 啊~文档也不太详细,比较简单,就查了下有没有其他更好的?
于是,发现了 codecov。
干净、免费,我喜欢。
文档也相对于 coveralls 更清晰、详细。在尝试之后,更是觉得我的选择是明智的。^_^
codecov 的使用相当简单,甚至不用看文档就可以轻易配置。
首先,登录首页,根据自己源码的存储位置选择相应的登录按钮,这里我选择 Github,第一次登录会需要你的授权。
授权成功之后,就能看到类似下面的图,分别对应你的个人账户以及你所加入的组织。
第一次使用时,默认是没有 repository 的,需要通过点击 + Add my first repository
来添加需要 codecov 管理的 repository。
选择相应的 repository 之后,你可以看到一个类似下面的页面。当然,数据什么肯定是没有的。
前几个 tab 是用来展示信息的,在配置完成并运行之前是没有信息的,配置的时候只需要看最后一个 setting tab。
切换左侧的菜单,就能分别看到 setting 和 badge 的信息,是不是超级赞?
无论 codecov 还是 coveralls,它自身都不会去运行测试用例来获得项目代码的覆盖率,而是通过收集覆盖率报告及其他关键信息来静态分析。
codecov 可以接收 lcov, gcov 以及正确的 json 数据格式作为输入信息。
于是,如果你使用 JEST 作为测试框架,并开启测试覆盖率(collectCoverage),由于,JEST 使用 istanbul 生成覆盖率报告,即 lcov。那么,上传报告就异常简单了。只需安装 codecov
npm install codecov --save-dev
然后,在 CI 执行之后,上传报告就行。比如,像这样
language: node_js
node_js:
- "6"
cache:
directories: node_modules
script:
- npm run test:coverage
# 这里我没有全局安装 codecov,所以要通过 npm 来运行 codecov
- npm run codecov
os:
- linux
- osx
这次的 badge 如何获取上面有写到,这里就不再展示了。
跨浏览器测试同样有 2 个选择,这次我同三剑客的作者站在了同一战线,选择使用 SAUCELABS。
SAUCELABS 开源免费账号注册方式隐藏得比较好,找不到的可以点这里。
不过,由于 JEST 不支持 end-to-end 测试,所以,为了做跨浏览器测试我们不得不寻求其他的测试框架来帮助完成这一工作。这里我并不打算使用 karma,即使是 karma 同 SAUCELABS 有现成的集成插件 karma-sauce-launcher 可以使用。
不要问我为什么,就是这么任(jue)性(jiang)。
你真不问么?那我就说了吧。因为现有的测试框架 JEST 已经可以完成 karma 的大部分工作,单纯为 end-to-end 测试单独引入 karma 就没有必要了。
经过一番资料收集和比较之后,我选择 Nightwatch 来解决跨浏览器测试的问题。
What's Nightwatch?
Nightwatch.js is an automated testing framework for web applications and websites, written in Node.js and using the W3C WebDriver API (formerly Selenium WebDriver).
It is a complete browser (End-to-End) testing solution which aims to simplify the process of setting up Continuous Integration and writing automated tests.
可以从官网的介绍中看到,Nightwatch 对我们当前想解决的问题简直是正中下怀啊!(如果你的项目使用的是 Angular,那么,你也可以试试 Protractor)
在查资料时,发现 nightwatch 的第一个 issue 竟然是尤大大提的。
走得越远,越是发现一路都是大大们留下的足迹。
膜拜大大。
回到正题,使用 nightwatch 建立 e2e 测试也是相当容易的,这里就简要说一下流程。
首先,使用 npm 进行安装,这就不多说了。
然后,在根目录下添加配置文件,可以是 nightwatch.conf.js,也可以是 nightwatch.json。
接着,写对应的测试,API 参考官网。
最后,跑测试命令就好了。
主要是来看看,怎么将 nightwatch 的测试同 saucelabs 以及 travis-ci 整合到一起。先看看测试文件。
// nightwatch.conf.js
module.exports = {
src_folders: ['tests/e2e'], // 测试文件目录
output_folder: 'tests/reports', // 测试报告地址
custom_commands_path: 'tests/saucelabs', // 自定义命令,这里用来更新测试信息到 saucelabs
custom_assertions_path: '',
page_objects_path: '',
globals_path: '',
test_workers: {
enabled: true,
workers: 'auto'
},
test_settings: {
default: {
launch_url: 'http://localhost:8080', // 目标地址,用于测试中读取
selenium_port: 4445, // selenium server 的端口(selenium server 由 saucelabs 提供)
selenium_host: 'localhost', // selenium server 的地址(selenium server 由 saucelabs 提供)
username: process.env.SAUCE_USERNAME,
access_key: process.env.SAUCE_ACCESS_KEY,
silent: true,
screenshots: {
enabled: false,
path: ''
},
globals: {
waitForConditionTimeout: 15000
},
// 以下重要!!!
desiredCapabilities: {
build: `build-${process.env.TRAVIS_JOB_NUMBER}`,
public: 'public',
'tunnel-identifier': process.env.TRAVIS_JOB_NUMBER
}
},
// 以下是不同环境的配置
chrome: {
desiredCapabilities: {
browserName: 'chrome'
}
},
firefox: {
desiredCapabilities: {
browserName: 'firefox'
}
},
internet_explorer_10: {
desiredCapabilities: {
browserName: 'internet explorer',
version: '10'
}
},
internet_explorer_11: {
desiredCapabilities: {
browserName: 'internet explorer',
version: '11'
}
},
edge: {
desiredCapabilities: {
browserName: 'MicrosoftEdge'
}
}
}
};
这里要注意以下几点:(重要!!!这些折磨了我近一周)
运行 localhost 测试,要开启 sauce connect
开启 sauce connect 之后,设置运行环境 selenium_port: 4445
, selenium_host: 'localhost'
以上几点是本地测试时需注意的,下面是连通 travis 时需注意的:
配置 'tunnel-identifier': process.env.TRAVIS_JOB_NUMBER
,其中 process.env.TRAVIS_JOB_NUMBER
是 travis 运行时的全局变量
配置 process.env.SAUCE_USERNAME
和 process.env.SAUCE_ACCESS_KEY
,后面细讲
配置 build
和 public
属性,分别用于标识测试和查看权限,这两点对最后生成 browser matrix badge 有用,这两点在三剑客的文章中也有提到
配置好了 nightwatch 同 saucelabs,再修改下 travis 的配置,将 saucelabs 整合进去。
// .travis.yml
language: node_js
node_js:
- '6'
cache:
directories: node_modules
# 用于打包,并在 travis 上启动本地服务,用于 e2e test
before_script:
- npm run build
- node server.js &
script:
- npm run test:coverage
- npm run codecov
- npm run test:e2e
os:
- linux
- osx
env:
global:
- secure: v6CRj4CKMqxEQ9MSYKAkbmrBgIBZvoppICx6JyjQXhexPOVQKBvboCgdL0lOOZdGZ9rEqSMXvud97kBAFYd1sdP/kSwXdUct5BOMIT3a5GLtY5aQfOocBwR6IvmZpO2U+4VhrCwkzdaq2Ehq0fAXF1pkxDj9YkJZmwDNhTdfDGkib+AwDyr4TLQFC1QrD/4vmrULb3NZdW1KadFYjLzVF8FMa2tDSYMFFVymYu5nuCa/Z0dqSfFy8McYwBMzThDkDRHMT/sf4zKDPyxUwN7xGfC6T88xzCEaltN6K7MGMGKvl7Y0p7VjYW/+rO38936kj6xuPU6J7Vh2yKPJhhT2LtM7ucuo0XSpIxCxaKXWeEmYl2KkCMWNHgrWACE//WBFRNx/JQHimw+abr1Zt/3V9QmSEvnB3hHB0NQgJ2nVrVDjk51RSVaiP4sfQ8GVqEwr1+wJqe4wz7fV+jvRB9uUGgGsjsBbZi6ZycoMtOBoJ+miviRCjZvf9sOZKfIDjcuE5vETQcE37d/++yplCG0N83Kx+q67mbWXirfNj2CfXp7pwHTN+n21v1BSicXqQ6+jaNzD/pcN/GTHgZ5A+VkdcjSmEziuQTO035i1nnCB9TQdFeRdGdfo6DAiq8YOfyVkQ1lml6lWqbPqa4QWokRUD2yA/hAIzNWe5BeLF2JFQBc=
- secure: S0vWVM74eiAHhk+kqqvym9aIgqaaGyGz9H3rfmEZoG4iuvXjXRaHOOSHxIRVsh5RYXr0PWHAj24fpN5AyUOlu5NQiwACBqmpw9KZBgVekWFshA5uYmpNpCG9w5/UAQa9q2+EcndOCM4lAyuT2wVJ5WfsHRzIA5jUpK1YmUYtuVICTSkumRoEaxfPkwzcGLF7f6aP7mG1YRKeO1F9+RhBfaGN1kYordxIk/fniH8OFB0XiLZ5OIovaAIYFKic0P1wUFwa78jU2fovdObS8JySl2LP19eaLX0MgAFoPB7oLFPxFBN7FCID41TEodDdZtcNnKJT4uQ/iWRqww2BOwVQM9whyBTg8J4kJZALicR4CzGCuUbdyQd2kh/hNZ9d9SKb6YXdcZElFmh3FY6zgfgv5PAx+jDlkfzmgBh7OD5OM4GVrsCsjnaAlmTUNtRPx9B4ps0gbr25F1PxuNy+MXfwSYJdliL+N01BTpiGyts/EXAraWvEm5YkhWfTnbgc8osd3cX9vwB0QHksK+BpkaEs6XCwU6kGMxAJIlafRv6RslREdTPBpYaXB4sGqdYXWY+YFqNxsAwTB3KWIq/uhZmSkou1jZfZa2QonMuVot68U11U7afmPzX8KOVeO2IEcUjt6I4eCYQ+31xO/wSLIQ1uoRySQ2S9VCzr+yzDpu0KVps=
addons:
sauce_connect: true
你肯定会诧异 global
下面的那两串长