mirror of
https://github.com/koel/koel
synced 2024-12-23 19:13:06 +00:00
197 lines
5.4 KiB
TypeScript
197 lines
5.4 KiB
TypeScript
import isMobile from 'ismobilejs'
|
|
import { isObject, mergeWith } from 'lodash'
|
|
import type { RenderOptions } from '@testing-library/vue'
|
|
import { cleanup, createEvent, fireEvent, render } from '@testing-library/vue'
|
|
import { afterEach, beforeEach, vi } from 'vitest'
|
|
import { defineComponent, nextTick } from 'vue'
|
|
import { commonStore, userStore } from '@/stores'
|
|
import { http } from '@/services'
|
|
import factory from '@/__tests__/factory'
|
|
import { DialogBoxKey, MessageToasterKey, OverlayKey, RouterKey } from '@/symbols'
|
|
import { DialogBoxStub, MessageToasterStub, OverlayStub } from '@/__tests__/stubs'
|
|
import { routes } from '@/config'
|
|
import Router from '@/router'
|
|
import userEvent from '@testing-library/user-event'
|
|
import type { UserEvent } from '@testing-library/user-event/dist/types/setup/setup'
|
|
import type { EventType } from '@testing-library/dom/types/events'
|
|
|
|
// A deep-merge function that
|
|
// - supports symbols as keys (_.merge doesn't)
|
|
// - supports Vue's Ref type without losing reactivity (deepmerge doesn't)
|
|
// Credit: https://stackoverflow.com/a/60598589/794641
|
|
const deepMerge = (first: object, second: object) => {
|
|
return mergeWith(first, second, (a, b) => {
|
|
if (!isObject(b)) {
|
|
return b
|
|
}
|
|
|
|
// @ts-ignore
|
|
return Array.isArray(a) ? [...a, ...b] : { ...a, ...b }
|
|
})
|
|
}
|
|
|
|
const setPropIfNotExists = (obj: object | null, prop: any, value: any) => {
|
|
if (!obj) {
|
|
return
|
|
}
|
|
|
|
if (!Object.prototype.hasOwnProperty.call(obj, prop)) {
|
|
obj[prop] = value
|
|
}
|
|
}
|
|
|
|
export default abstract class UnitTestCase {
|
|
protected router: Router
|
|
protected user: UserEvent
|
|
private backupMethods = new Map()
|
|
|
|
public constructor () {
|
|
this.router = new Router(routes)
|
|
this.mock(http, 'request') // prevent actual HTTP requests from being made
|
|
this.user = userEvent.setup({ delay: null }) // @see https://github.com/testing-library/user-event/issues/833
|
|
|
|
this.setReadOnlyProperty(navigator, 'clipboard', {
|
|
writeText: vi.fn(),
|
|
})
|
|
|
|
this.beforeEach()
|
|
this.afterEach()
|
|
this.test()
|
|
}
|
|
|
|
protected beforeEach (cb?: Closure) {
|
|
beforeEach(() => {
|
|
commonStore.state.song_length = 10
|
|
commonStore.state.allows_download = true
|
|
commonStore.state.uses_i_tunes = true
|
|
commonStore.state.supports_batch_downloading = true
|
|
commonStore.state.supports_transcoding = true
|
|
cb && cb()
|
|
})
|
|
}
|
|
|
|
protected afterEach (cb?: Closure) {
|
|
afterEach(() => {
|
|
isMobile.any = false
|
|
commonStore.state.song_length = 10
|
|
cleanup()
|
|
this.restoreAllMocks()
|
|
this.disablePlusEdition()
|
|
cb && cb()
|
|
})
|
|
}
|
|
|
|
protected be (user?: User) {
|
|
userStore.state.current = user || factory('user')
|
|
return this
|
|
}
|
|
|
|
protected beAdmin () {
|
|
factory.states('admin')('user')
|
|
return this.be(factory.states('admin')('user'))
|
|
}
|
|
|
|
protected mock<T, M extends MethodOf<Required<T>>> (obj: T, methodName: M, implementation?: any) {
|
|
const mock = vi.fn()
|
|
|
|
if (implementation !== undefined) {
|
|
mock.mockImplementation(implementation instanceof Function ? implementation : () => implementation)
|
|
}
|
|
|
|
this.backupMethods.set([obj, methodName], obj[methodName])
|
|
|
|
// @ts-ignore
|
|
obj[methodName] = mock
|
|
|
|
return mock
|
|
}
|
|
|
|
protected restoreAllMocks () {
|
|
this.backupMethods.forEach((fn, [obj, methodName]) => (obj[methodName] = fn))
|
|
this.backupMethods = new Map()
|
|
}
|
|
|
|
protected render (component: any, options: RenderOptions = {}) {
|
|
return render(component, deepMerge({
|
|
global: {
|
|
directives: {
|
|
'koel-focus': {},
|
|
'koel-tooltip': {},
|
|
'koel-hide-broken-icon': {},
|
|
'koel-overflow-fade': {},
|
|
},
|
|
components: {
|
|
Icon: this.stub('Icon'),
|
|
},
|
|
},
|
|
}, this.supplyRequiredProvides(options)))
|
|
}
|
|
|
|
protected enablePlusEdition () {
|
|
commonStore.state.koel_plus = {
|
|
active: true,
|
|
short_key: '****-XXXX',
|
|
customer_name: 'John Doe',
|
|
customer_email: 'Koel Plus',
|
|
product_id: 'koel-plus',
|
|
}
|
|
|
|
return this
|
|
}
|
|
|
|
protected disablePlusEdition () {
|
|
commonStore.state.koel_plus = {
|
|
active: false,
|
|
short_key: '',
|
|
customer_name: '',
|
|
customer_email: '',
|
|
product_id: '',
|
|
}
|
|
|
|
return this
|
|
}
|
|
|
|
protected stub (testId = 'stub') {
|
|
return defineComponent({
|
|
template: `<br data-testid="${testId}"/>`,
|
|
})
|
|
}
|
|
|
|
protected async tick (count = 1) {
|
|
for (let i = 0; i < count; ++i) {
|
|
await nextTick()
|
|
}
|
|
}
|
|
|
|
protected setReadOnlyProperty<T> (obj: T, prop: keyof T, value: any) {
|
|
return Object.defineProperties(obj, {
|
|
[prop]: {
|
|
value,
|
|
configurable: true,
|
|
},
|
|
})
|
|
}
|
|
|
|
protected async type (element: HTMLElement, value: string) {
|
|
await this.user.clear(element)
|
|
await this.user.type(element, value)
|
|
}
|
|
|
|
protected async trigger (element: HTMLElement, key: EventType | string, options?: object = {}) {
|
|
await fireEvent(element, createEvent[key](element, options))
|
|
}
|
|
|
|
protected abstract test ()
|
|
|
|
private supplyRequiredProvides (options: RenderOptions) {
|
|
options.global = options.global || {}
|
|
options.global.provide = options.global.provide || {}
|
|
|
|
setPropIfNotExists(options.global.provide, DialogBoxKey, DialogBoxStub)
|
|
setPropIfNotExists(options.global.provide, MessageToasterKey, MessageToasterStub)
|
|
setPropIfNotExists(options.global.provide, OverlayKey, OverlayStub)
|
|
setPropIfNotExists(options.global.provide, RouterKey, this.router)
|
|
|
|
return options
|
|
}
|
|
}
|