Skip to content

Commit 207b027

Browse files
ryanthemanueljennifer-shehanemschile
authored
feat: support multiple servers from which to gather code coverage (#880)
Co-authored-by: Jennifer Shehane <[email protected]> Co-authored-by: Matt Schile <[email protected]>
1 parent 83bd4fe commit 207b027

File tree

13 files changed

+142
-26
lines changed

13 files changed

+142
-26
lines changed

.circleci/config.yml

+2
Original file line numberDiff line numberDiff line change
@@ -152,6 +152,7 @@ workflows:
152152
- exclude-files
153153
- frontend
154154
- fullstack
155+
- multiple-backends
155156
- one-spec
156157
- same-folder
157158
- support-files
@@ -179,6 +180,7 @@ workflows:
179180
- test-exclude-files
180181
- test-frontend
181182
- test-fullstack
183+
- test-multiple-backends
182184
- test-one-spec
183185
- test-same-folder
184186
- test-support-files

README.md

+12
Original file line numberDiff line numberDiff line change
@@ -214,6 +214,18 @@ if (global.__coverage__) {
214214
}
215215
```
216216

217+
Or if you have multiple servers from which you are wanting to gather code coverage, you can pass an array to `url` as well:
218+
219+
```json
220+
{
221+
"env": {
222+
"codeCoverage": {
223+
"url": ["http://localhost:3000/__coverage__", "http://localhost:3001/__coverage__"]
224+
}
225+
}
226+
}
227+
```
228+
217229
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.
218230

219231
### expectBackendCoverageOnly

support.js

+36-26
Original file line numberDiff line numberDiff line change
@@ -148,39 +148,49 @@ const registerHooks = () => {
148148
// we can only request server-side code coverage
149149
// if we are running end-to-end tests,
150150
// otherwise where do we send the request?
151-
const url = Cypress._.get(
151+
const captureUrls = Cypress._.get(
152152
Cypress.env('codeCoverage'),
153153
'url',
154154
'/__coverage__'
155155
)
156-
cy.request({
157-
url,
158-
log: false,
159-
failOnStatusCode: false
160-
})
161-
.then((r) => {
162-
return Cypress._.get(r, 'body.coverage', null)
156+
function captureCoverage(url, suffix = '') {
157+
cy.request({
158+
url,
159+
log: false,
160+
failOnStatusCode: false
163161
})
164-
.then((coverage) => {
165-
if (!coverage) {
166-
// we did not get code coverage - this is the
167-
// original failed request
168-
const expectBackendCoverageOnly = Cypress._.get(
169-
Cypress.env('codeCoverage'),
170-
'expectBackendCoverageOnly',
171-
false
172-
)
173-
if (expectBackendCoverageOnly) {
174-
throw new Error(
175-
`Expected to collect backend code coverage from ${url}`
162+
.then((r) => {
163+
return Cypress._.get(r, 'body.coverage', null)
164+
})
165+
.then((coverage) => {
166+
if (!coverage) {
167+
// we did not get code coverage - this is the
168+
// original failed request
169+
const expectBackendCoverageOnly = Cypress._.get(
170+
Cypress.env('codeCoverage'),
171+
'expectBackendCoverageOnly',
172+
false
176173
)
177-
} else {
178-
// we did not really expect to collect the backend code coverage
179-
return
174+
if (expectBackendCoverageOnly) {
175+
throw new Error(
176+
`Expected to collect backend code coverage from ${url}`
177+
)
178+
} else {
179+
// we did not really expect to collect the backend code coverage
180+
return
181+
}
180182
}
181-
}
182-
sendCoverage(coverage, 'backend')
183-
})
183+
sendCoverage(coverage, `backend${suffix}`)
184+
})
185+
}
186+
187+
if (Array.isArray(captureUrls)) {
188+
for (const [index, url] of captureUrls.entries()) {
189+
captureCoverage(url, `_${index}`)
190+
}
191+
} else {
192+
captureCoverage(captureUrls)
193+
}
184194
}
185195
})
186196

test-apps/multiple-backends/.babelrc

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
{
2+
"plugins": ["istanbul"]
3+
}

test-apps/multiple-backends/README.md

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
# example: multiple backends
2+
3+
> Getting code coverage from multiple backends
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
const { defineConfig } = require('cypress')
2+
3+
module.exports = defineConfig({
4+
fixturesFolder: false,
5+
env: {
6+
codeCoverage: {
7+
url: ['http://localhost:3003/__coverage__', 'http://localhost:3004/__coverage__'],
8+
expectBackendCoverageOnly: true,
9+
},
10+
},
11+
e2e: {
12+
setupNodeEvents(on, config) {
13+
return require('./cypress/plugins/index.js')(on, config)
14+
},
15+
baseUrl: 'http://localhost:3003',
16+
},
17+
})
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
/// <reference types="Cypress" />
2+
it('has multiple backends with code coverage', () => {
3+
cy.visit('/')
4+
cy.request('/hello')
5+
cy.request('http://localhost:3004/world')
6+
})
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
module.exports = (on, config) => {
2+
require('@cypress/code-coverage/task')(on, config)
3+
return config
4+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
import '@cypress/code-coverage/support'
+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
{
2+
"name": "example-multiple-backends",
3+
"description": "Code coverage for multiple backends",
4+
"private": true,
5+
"scripts": {
6+
"cy:run": "cypress run",
7+
"start": "nyc --silent node server/server-3003 & nyc --silent node server/server-3004",
8+
"pretest": "rimraf .nyc_output .cache coverage dist",
9+
"test": "start-test \"3003|3004\" cy:run",
10+
"coverage:verify": "npx nyc report --check-coverage true --lines 100",
11+
"coverage:check-files": "check-coverage server-3003.js server-3004.js && only-covered server-3003.js server-3004.js"
12+
}
13+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
<body>
2+
test multiple backends
3+
</body>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
const express = require('express')
2+
const app = express()
3+
const port = 3003
4+
5+
// if there is code coverage information
6+
// then expose an endpoint that returns it
7+
/* istanbul ignore next */
8+
if (global.__coverage__) {
9+
console.log('have code coverage, will add middleware for express')
10+
console.log(`to fetch: GET :${port}/__coverage__`)
11+
require('@cypress/code-coverage/middleware/express')(app)
12+
}
13+
14+
app.use(express.static(__dirname))
15+
16+
app.get('/hello', (req, res) => {
17+
console.log('sending hello')
18+
res.send('Hello')
19+
})
20+
21+
app.listen(port, () => console.log(`Example app listening on port ${port}!`))
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
const express = require('express')
2+
const app = express()
3+
const port = 3004
4+
5+
// if there is code coverage information
6+
// then expose an endpoint that returns it
7+
/* istanbul ignore next */
8+
if (global.__coverage__) {
9+
console.log('have code coverage, will add middleware for express')
10+
console.log(`to fetch: GET :${port}/__coverage__`)
11+
require('@cypress/code-coverage/middleware/express')(app)
12+
}
13+
14+
app.use(express.static(__dirname))
15+
16+
app.get('/world', (req, res) => {
17+
console.log('sending world')
18+
res.send('World!')
19+
})
20+
21+
app.listen(port, () => console.log(`Example app listening on port ${port}!`))

0 commit comments

Comments
 (0)