I am running a vue3 application using the Composition API and the setup()
hook.
I am using Vitest as unit-test framework. (v 0.6.1)
I have the following sample ponent :
// src/ponents/MyComponent.vue
<template>
<div>
<h1>counter : {{ counter }}</h1>
<button
@click="incrementCounter"
>
Click
</button>
</div>
</template>
<script setup lang="ts">
// imports
import { ref } from 'vue'
// datas
const counter = ref(1)
// methods
const incrementCounter = () => {
if (confirm()) { // call the confirm method
counter.value++ // increment counter by 1
}
}
const confirm = () => {
return true
}
</script>
And its test file :
// src/ponents/MyComponent.spec.ts
import {
shallowMount
} from '@vue/test-utils'
import MyComponent from '@/ponents/MyComponent.vue'
describe('ponent/MyComponent.vue', () => {
it('incrementCounter method', () => {
const wrapper = shallowMount(MyComponent) // create the wrapper
const confirmSpy = vi.spyOn(wrapper.vm, 'confirm') // create the confirm method spy
wrapper.vm.incrementCounter() // use the incrementCounter method
expect(wrapper.vm.counter).toBe(2) // test passed
expect(confirmSpy).toHaveBeenCalled() // test failed
})
})
The goal of the test is simply to verify if the confirm()
method has been called inside the incrementCounter()
method or not.
I tried to use the vitest tohavebeencalled() method with a spy of the confirm()
method but the test end up in failure with the following message:
Re-running tests... [ src/ponents/MyComponent.spec.ts ]
× src/ponents/MyComponent.spec.ts > ponent/MyComponent.vue > incrementCounter method → expected "confirm" to be called at least once
⎯⎯⎯⎯⎯⎯⎯ Failed Tests 1 ⎯⎯⎯⎯⎯⎯⎯
FAIL src/ponents/MyComponent.spec.ts > ponent/MyComponent.vue
incrementCounter method AssertionError: expected "confirm" to be called at least once ❯ src/ponents/MyComponent.spec.ts:13:23 11| wrapper.vm.incrementCounter() // use the incrementCounter method 12| expect(wrapper.vm.counter).toBe(2) // test passed 13| expect(confirmSpy).toHaveBeenCalled() // test failed | ^ 14| }) 15| })
It seems to indicate that confirm()
method has not been called, but since the counter value has been increased to 2, I guess it implies that the method has been effectively called in fact.
I am using the spyOn()
method wrong? What should I do to make this test pass?
Thank you in advance for your help.
I am running a vue3 application using the Composition API and the setup()
hook.
I am using Vitest as unit-test framework. (v 0.6.1)
I have the following sample ponent :
// src/ponents/MyComponent.vue
<template>
<div>
<h1>counter : {{ counter }}</h1>
<button
@click="incrementCounter"
>
Click
</button>
</div>
</template>
<script setup lang="ts">
// imports
import { ref } from 'vue'
// datas
const counter = ref(1)
// methods
const incrementCounter = () => {
if (confirm()) { // call the confirm method
counter.value++ // increment counter by 1
}
}
const confirm = () => {
return true
}
</script>
And its test file :
// src/ponents/MyComponent.spec.ts
import {
shallowMount
} from '@vue/test-utils'
import MyComponent from '@/ponents/MyComponent.vue'
describe('ponent/MyComponent.vue', () => {
it('incrementCounter method', () => {
const wrapper = shallowMount(MyComponent) // create the wrapper
const confirmSpy = vi.spyOn(wrapper.vm, 'confirm') // create the confirm method spy
wrapper.vm.incrementCounter() // use the incrementCounter method
expect(wrapper.vm.counter).toBe(2) // test passed
expect(confirmSpy).toHaveBeenCalled() // test failed
})
})
The goal of the test is simply to verify if the confirm()
method has been called inside the incrementCounter()
method or not.
I tried to use the vitest tohavebeencalled() method with a spy of the confirm()
method but the test end up in failure with the following message:
Re-running tests... [ src/ponents/MyComponent.spec.ts ]
× src/ponents/MyComponent.spec.ts > ponent/MyComponent.vue > incrementCounter method → expected "confirm" to be called at least once
⎯⎯⎯⎯⎯⎯⎯ Failed Tests 1 ⎯⎯⎯⎯⎯⎯⎯
FAIL src/ponents/MyComponent.spec.ts > ponent/MyComponent.vue
incrementCounter method AssertionError: expected "confirm" to be called at least once ❯ src/ponents/MyComponent.spec.ts:13:23 11| wrapper.vm.incrementCounter() // use the incrementCounter method 12| expect(wrapper.vm.counter).toBe(2) // test passed 13| expect(confirmSpy).toHaveBeenCalled() // test failed | ^ 14| }) 15| })
It seems to indicate that confirm()
method has not been called, but since the counter value has been increased to 2, I guess it implies that the method has been effectively called in fact.
I am using the spyOn()
method wrong? What should I do to make this test pass?
Thank you in advance for your help.
Share Improve this question edited May 9, 2022 at 4:11 tony19 139k23 gold badges277 silver badges347 bronze badges asked May 2, 2022 at 8:42 DeltantoineDeltantoine 611 gold badge1 silver badge2 bronze badges 2- I think this can help you: stackoverflow./questions/70390151/… – DnD2k21 Commented May 2, 2022 at 8:47
-
1
Thank you for the hint ! It's actually possible to achiveve the test like that :
wrapper.vm.incrementCounter = vi.fn(() => wrapper.vm.confirm());
vi.spyOn(wrapper.vm, 'confirm');
wrapper.vm.incrementCounter();
expect(wrapper.vm.confirm).toHaveBeenCalled();
But it's not super handy, specially when you working with promises and async functions. Using a posable function seems to be a better option. – Deltantoine Commented May 11, 2022 at 1:10
1 Answer
Reset to default 9In this case, the reference to confirm
in incrementCounter()
cannot be externally modified. That's as impossible as modifying a function's private variables.
Here's a similar vanilla JavaScript example that demonstrates what the code in your <script setup>
is attempting (run the snippet below for demo):
function setup() {
const incrementCounter = () => {
confirm()
}
const confirm = () => {
console.log('confirm')
}
return {
incrementCounter,
confirm,
}
}
const p = setup()
p.incrementCounter()
console.log('trying to update confirm...')
p.confirm = () => console.log('Yes do it!')
p.incrementCounter() // ❌ still calls original confirm
However, the properties of an object can be modified externally. So, a workaround is to update incrementCounter()
to reference confirm
via an object, which we'll later update:
function setup() {
const incrementCounter = () => {
/*