Skip to content

Commit 6cc65fc

Browse files
authored
feat: add server-side middleware and merge the backend coverage with frontend (#22)
* chore: add middleware and merge server-side coverage * chore: configure coverage url * finish docs and add link to the realworld repo
1 parent dccb126 commit 6cc65fc

File tree

4 files changed

+86
-0
lines changed

4 files changed

+86
-0
lines changed

.vscode/settings.json

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
{
2+
"workbench.colorTheme": "Tomorrow Night Blue"
3+
}

README.md

+55
Original file line numberDiff line numberDiff line change
@@ -63,10 +63,65 @@ module.exports = (on, config) => {
6363

6464
Now the code coverage from spec files will be combined with end-to-end coverage.
6565

66+
## Instrument backend code
67+
68+
You can also instrument your server-side code and produce combined coverage report that covers both the backend and frontend code.
69+
70+
1. Run the server code with instrumentation. The simplest way is to use [nyc](https://github.com/istanbuljs/nyc). If normally you run `node src/server` then to run instrumented version you can do `nyc --silent node src/server`.
71+
2. Add an endpoint that returns collected coverage. If you are using Express, you can simply do
72+
73+
```js
74+
const express = require('express')
75+
const app = express()
76+
require('@cypress/code-coverage/middleware')(app)
77+
```
78+
79+
**Tip:** you can register the endpoint only if there is global code coverage object, and you can exclude the middleware code from the coverage numbers
80+
81+
```js
82+
// https://github.com/gotwarlost/istanbul/blob/master/ignoring-code-for-coverage.md
83+
/* istanbul ignore next */
84+
if (global.__coverage__) {
85+
require('@cypress/code-coverage/middleware')(app)
86+
}
87+
```
88+
89+
If you use Hapi server, define the endpoint yourself and return the object
90+
91+
```js
92+
// https://github.com/gotwarlost/istanbul/blob/master/ignoring-code-for-coverage.md
93+
/* istanbul ignore next */
94+
if (global.__coverage__) {
95+
// https://hapijs.com/tutorials/routing?lang=en_US
96+
server.route({
97+
method: 'GET',
98+
path: '/__coverage__',
99+
handler () {
100+
return { coverage: global.__coverage__ }
101+
}
102+
})
103+
}
104+
```
105+
106+
3. Save the API coverage endpoint in `cypress.json` file to let the plugin know where to call to receive the code coverage data from the server. Place it in `env.codeCoverage` object:
107+
108+
```json
109+
{
110+
"env": {
111+
"codeCoverage": {
112+
"url": "http://localhost:3000/__coverage__"
113+
}
114+
}
115+
}
116+
```
117+
118+
That should be enough - the code coverage from the server will be requested at the end of the test run and merged with the client-side code coverage, producing a combined report
119+
66120
## Examples
67121

68122
- [Cypress code coverage guide](http://on.cypress.io/code-coverage)
69123
- [cypress-example-todomvc-redux](https://github.com/cypress-io/cypress-example-todomvc-redux)
124+
- Full frontend + backend code coverage in [bahmutov/realworld](https://github.com/bahmutov/realworld) repo
70125
- Read ["Code Coverage by Parcel Bundler"](https://glebbahmutov.com/blog/code-coverage-by-parcel/) blog post
71126
- Read ["Combined End-to-end and Unit Test Coverage"](https://glebbahmutov.com/blog/combined-end-to-end-and-unit-test-coverage/)
72127

middleware.js

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
module.exports = app => {
2+
// expose "GET __coverage__" endpoint that just returns
3+
// global coverage information (if the application has been instrumented)
4+
app.get('/__coverage__', (req, res) => {
5+
res.json({
6+
coverage: global.__coverage__ || null
7+
})
8+
})
9+
}

support.js

+19
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,25 @@ afterEach(() => {
2121
})
2222

2323
after(() => {
24+
// there might be server-side code coverage information
25+
// we should grab it once after all tests finish
26+
const url = Cypress._.get(Cypress.env('codeCoverage'), 'url', '/__coverage__')
27+
cy.request({
28+
url,
29+
log: false,
30+
failOnStatusCode: false
31+
})
32+
.then(r => Cypress._.get(r, 'body.coverage', null), { log: false })
33+
.then(coverage => {
34+
if (!coverage) {
35+
// we did not get code coverage - this is the
36+
// original failed request
37+
return
38+
}
39+
cy.task('combineCoverage', coverage)
40+
})
41+
42+
// collect and merge frontend coverage
2443
const specFolder = Cypress.config('integrationFolder')
2544
const supportFolder = Cypress.config('supportFolder')
2645

0 commit comments

Comments
 (0)