Skip to content

Commit

Permalink
Fix tests to cover uncovered branches
Browse files Browse the repository at this point in the history
This commit solves the coverage drop from:

- #23
  01a79f6

Restores `istanbul` as the coverage reporter for the JSDom-based CI run
to restore consistency in reporting, particularly in Coveralls.

---

Adapted from the new test/missing-app-div/jsdom.test.js:

Moved the "missing #app div" test into a separate directory from the
other tests in ../jsdom.test.js. to prevent Node.js from using the same
`test-modules/main.js` import. Otherwise:

- If the test case were in the same file, the "missing #app div" branch
  wouldn't execute and the test would fail.

- If this test file were in the same directory, the Istanbul coverage
  reporter wouldn't see the coverage from the "missing #app div" branch.
  I don't know exactly why that is.

At the same time, the previous `src="./main.js?version=missing"` query
suffix is no longer necessary.

I got the idea to organize the tests this way after successfully
covering similar code in:

- mbland/tomcat-servlet-testing-example#85
  mbland/tomcat-servlet-testing-example@b5df30e
  • Loading branch information
mbland committed Jan 16, 2024
1 parent 1cba4d6 commit 93dd0cf
Show file tree
Hide file tree
Showing 6 changed files with 79 additions and 26 deletions.
1 change: 0 additions & 1 deletion ci/vitest.config.browser.js
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,6 @@ export default mergeConfig(baseConfig, defineConfig({
test: {
outputFile: 'TESTS-TestSuites-browser.xml',
coverage: {
provider: 'istanbul',
reportsDirectory: 'coverage-browser'
},
browser: {
Expand Down
1 change: 1 addition & 0 deletions ci/vitest.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ export default mergeConfig(viteConfig, defineConfig({
reporters: [ 'junit', 'default' ],
coverage: {
enabled: true,
provider: 'istanbul',
reporter: [ 'text', 'lcovonly' ],
reportsDirectory: 'coverage-jsdom'
}
Expand Down
5 changes: 1 addition & 4 deletions test-modules/missing.html
Original file line number Diff line number Diff line change
Expand Up @@ -7,10 +7,7 @@
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Test Page for JsdomPageOpener missing an app div</title>
<!-- Provide a query suffix so import from index.html doesn't get reused and
- cause "JsdomPageOpener > doesn't throw if missing app div" to fail.
- https://github.com/nodejs/help/issues/2751#issuecomment-1075535742 -->
<script type="module" src="./main.js?version=missing"></script>
<script type="module" src="./main.js"></script>
</head>
<body>
<div id="not-the-div-you're-looking-for"></div>
Expand Down
15 changes: 15 additions & 0 deletions test/jsdom-fixture.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
/*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at https://mozilla.org/MPL/2.0/.
*/

export default class JsdomFixture {
static #origWindow = globalThis.window
static #origDocument = globalThis.document

restoreOrigWindowAndDocument() {
globalThis.window = JsdomFixture.#origWindow
globalThis.document = JsdomFixture.#origDocument
}
}
26 changes: 5 additions & 21 deletions test/jsdom.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -6,22 +6,17 @@
*/

import JsdomPageOpener from '../lib/jsdom.js'
import { afterEach, beforeAll, describe, expect, test, vi } from 'vitest'

describe.skipIf(globalThis.window !== undefined)('JsdomPageOpener', () => {
const origWindow = globalThis.window
const origDocument = globalThis.document
import JsdomFixture from './jsdom-fixture.js'
import { afterEach, beforeAll, describe, expect, test } from 'vitest'

describe.skipIf(globalThis.window)('JsdomPageOpener', () => {
/** @type {JsdomPageOpener} */
let opener
const fixture = new JsdomFixture()

beforeAll(async () => {opener = new JsdomPageOpener(await import('jsdom'))})

afterEach(() => {
globalThis.window = origWindow
globalThis.document = origDocument
vi.restoreAllMocks()
})
afterEach(fixture.restoreOrigWindowAndDocument)

test('restores original globalThis.{window,document}', async () => {
const pagePath = './test-modules/index.html'
Expand Down Expand Up @@ -50,15 +45,4 @@ describe.skipIf(globalThis.window !== undefined)('JsdomPageOpener', () => {
expect(err.cause.cause.message).toBe('test error')
})
})

test('doesn\'t throw if missing app div', async () => {
const pagePath = './test-modules/missing.html'
const consoleSpy = vi.spyOn(console, 'error')
.mockImplementationOnce(() => {})

const { close } = await opener.open('/basedir/', pagePath)
close()

expect(consoleSpy).toBeCalledWith('no #app element')
})
})
57 changes: 57 additions & 0 deletions test/missing-app-div/jsdom.test.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
/* eslint-env browser */
/*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at https://mozilla.org/MPL/2.0/.
*/

import JsdomPageOpener from '../../lib/jsdom.js'
import JsdomFixture from '../jsdom-fixture.js'
import { afterEach, beforeAll, describe, expect, test, vi } from 'vitest'

// This is in a separate directory from the other tests in ../jsdom.test.js. to
// prevent Node.js from using the same `test-modules/main.js` import. Otherwise:
//
// - If the test case were in the same file, the "missing #app div" branch
// wouldn't execute and the test would fail.
//
// - If this test file were in the same directory, the Istanbul coverage
// reporter wouldn't see the coverage from the "missing #app div" branch. I
// don't know exactly why that is.
//
// At the same time, the previous `src="./main.js?version=missing"` query suffix
// is no longer necessary.
//
// This solves the coverage drop from:
//
// - mbland/test-page-opener#23
// mbland/test-page-opener@01a79f6
//
// I got the idea to organize the tests this way after successfully covering
// similar code in:
//
// - mbland/tomcat-servlet-testing-example#85
// mbland/tomcat-servlet-testing-example@b5df30e
describe.skipIf(globalThis.window)('JsdomPageOpener', () => {
/** @type {JsdomPageOpener} */
let opener
const fixture = new JsdomFixture()

beforeAll(async () => {opener = new JsdomPageOpener(await import('jsdom'))})

afterEach(() => {
fixture.restoreOrigWindowAndDocument()
vi.restoreAllMocks()
})

test('logs error if missing #app div', async () => {
const pagePath = './test-modules/missing.html'
const consoleSpy = vi.spyOn(console, 'error')
.mockImplementationOnce(() => {})

const { close } = await opener.open('/basedir/', pagePath)
close()

expect(consoleSpy).toBeCalledWith('no #app element')
})
})

0 comments on commit 93dd0cf

Please sign in to comment.