最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - Vue3- Using Vitest toHaveBeenCalled() method - Stack Overflow

programmeradmin0浏览0评论

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
Add a ment  | 

1 Answer 1

Reset to default 9

In 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 = () => {
   /*
发布评论

评论列表(0)

  1. 暂无评论