mirror of
https://github.com/langgenius/dify.git
synced 2026-05-08 11:47:35 +08:00
Signed-off-by: majiayu000 <1835304752@qq.com> Signed-off-by: dependabot[bot] <support@github.com> Signed-off-by: NeatGuyCoding <15627489+NeatGuyCoding@users.noreply.github.com> Signed-off-by: -LAN- <laipz8200@outlook.com> Signed-off-by: yihong0618 <zouzou0208@gmail.com> Co-authored-by: QuantumGhost <obelisk.reg+git@gmail.com> Co-authored-by: 盐粒 Yanli <yanli@dify.ai> Co-authored-by: wangxiaolei <fatelei@gmail.com> Co-authored-by: Stephen Zhou <38493346+hyoban@users.noreply.github.com> Co-authored-by: gemini-code-assist[bot] <176961590+gemini-code-assist[bot]@users.noreply.github.com> Co-authored-by: Cursx <33718736+Cursx@users.noreply.github.com> Co-authored-by: autofix-ci[bot] <114827586+autofix-ci[bot]@users.noreply.github.com> Co-authored-by: lif <1835304752@qq.com> Co-authored-by: 非法操作 <hjlarry@163.com> Co-authored-by: Asuka Minato <i@asukaminato.eu.org> Co-authored-by: fenglin <790872612@qq.com> Co-authored-by: qiaofenglin <qiaofenglin@baidu.com> Co-authored-by: -LAN- <laipz8200@outlook.com> Co-authored-by: TomoOkuyama <49631611+TomoOkuyama@users.noreply.github.com> Co-authored-by: Tomo Okuyama <tomo.okuyama@intersystems.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: zyssyz123 <916125788@qq.com> Co-authored-by: hj24 <mambahj24@gmail.com> Co-authored-by: Coding On Star <447357187@qq.com> Co-authored-by: CodingOnStar <hanxujiang@dify.ai> Co-authored-by: yyh <92089059+lyzno1@users.noreply.github.com> Co-authored-by: Xiangxuan Qu <fghpdf@outlook.com> Co-authored-by: fghpdf <fghpdf@users.noreply.github.com> Co-authored-by: coopercoder <whitetiger0127@163.com> Co-authored-by: zhaiguangpeng <zhaiguangpeng@didiglobal.com> Co-authored-by: Junyan Qin (Chin) <rockchinq@gmail.com> Co-authored-by: E.G <146701565+GlobalStar117@users.noreply.github.com> Co-authored-by: GlobalStar117 <GlobalStar117@users.noreply.github.com> Co-authored-by: Claude Haiku 4.5 <noreply@anthropic.com> Co-authored-by: CodingOnStar <hanxujiang@dify.com> Co-authored-by: crazywoola <100913391+crazywoola@users.noreply.github.com> Co-authored-by: heyszt <270985384@qq.com> Co-authored-by: NeatGuyCoding <15627489+NeatGuyCoding@users.noreply.github.com> Co-authored-by: Yeuoly <45712896+Yeuoly@users.noreply.github.com> Co-authored-by: zxhlyh <jasonapring2015@outlook.com> Co-authored-by: moonpanda <chuanzegao@163.com> Co-authored-by: warlocgao <warlocgao@tencent.com> Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com> Co-authored-by: claude[bot] <41898282+claude[bot]@users.noreply.github.com> Co-authored-by: KVOJJJin <jzongcode@gmail.com> Co-authored-by: eux <euxx@users.noreply.github.com> Co-authored-by: bangjiehan <bangjiehan@gmail.com> Co-authored-by: FFXN <31929997+FFXN@users.noreply.github.com> Co-authored-by: Jyong <76649700+JohnJyong@users.noreply.github.com> Co-authored-by: Nie Ronghua <nieronghua@sf-express.com> Co-authored-by: JQSevenMiao <141806521+JQSevenMiao@users.noreply.github.com> Co-authored-by: jiasiqi <jiasiqi3@tal.com> Co-authored-by: Seokrin Taron Sung <sungsjade@gmail.com> Co-authored-by: CrabSAMA <40541269+CrabSAMA@users.noreply.github.com> Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: yihong <zouzou0208@gmail.com> Co-authored-by: Joel <iamjoel007@gmail.com> Co-authored-by: Wu Tianwei <30284043+WTW0313@users.noreply.github.com> Co-authored-by: yessenia <yessenia.contact@gmail.com> Co-authored-by: Jax <anobaka@qq.com> Co-authored-by: niveshdandyan <155956228+niveshdandyan@users.noreply.github.com> Co-authored-by: OSS Contributor <oss-contributor@example.com> Co-authored-by: niveshdandyan <niveshdandyan@users.noreply.github.com> Co-authored-by: Sean Kenneth Doherty <Smaster7772@gmail.com>
159 lines
4.9 KiB
TypeScript
159 lines
4.9 KiB
TypeScript
import { act, cleanup } from '@testing-library/react'
|
|
import { mockAnimationsApi, mockResizeObserver } from 'jsdom-testing-mocks'
|
|
import '@testing-library/jest-dom/vitest'
|
|
|
|
mockResizeObserver()
|
|
|
|
// Mock Web Animations API for Headless UI
|
|
mockAnimationsApi()
|
|
|
|
// Suppress act() warnings from @headlessui/react internal Transition component
|
|
// These warnings are caused by Headless UI's internal async state updates, not our code
|
|
const originalConsoleError = console.error
|
|
console.error = (...args: unknown[]) => {
|
|
// Check all arguments for the Headless UI TransitionRootFn act warning
|
|
const fullMessage = args.map(arg => (typeof arg === 'string' ? arg : '')).join(' ')
|
|
if (fullMessage.includes('TransitionRootFn') && fullMessage.includes('not wrapped in act'))
|
|
return
|
|
originalConsoleError.apply(console, args)
|
|
}
|
|
|
|
// Fix for @headlessui/react compatibility with happy-dom
|
|
// headlessui tries to override focus properties which may be read-only in happy-dom
|
|
if (typeof window !== 'undefined') {
|
|
// Provide a minimal animations API polyfill before @headlessui/react boots
|
|
if (typeof Element !== 'undefined' && !Element.prototype.getAnimations)
|
|
Element.prototype.getAnimations = () => []
|
|
|
|
if (!document.getAnimations)
|
|
document.getAnimations = () => []
|
|
|
|
const ensureWritable = (target: object, prop: string) => {
|
|
const descriptor = Object.getOwnPropertyDescriptor(target, prop)
|
|
if (descriptor && !descriptor.writable) {
|
|
const original = descriptor.value ?? descriptor.get?.call(target)
|
|
Object.defineProperty(target, prop, {
|
|
value: typeof original === 'function' ? original : vi.fn(),
|
|
writable: true,
|
|
configurable: true,
|
|
})
|
|
}
|
|
}
|
|
|
|
ensureWritable(window, 'focus')
|
|
ensureWritable(HTMLElement.prototype, 'focus')
|
|
}
|
|
|
|
if (typeof globalThis.ResizeObserver === 'undefined') {
|
|
globalThis.ResizeObserver = class {
|
|
observe() {
|
|
return undefined
|
|
}
|
|
|
|
unobserve() {
|
|
return undefined
|
|
}
|
|
|
|
disconnect() {
|
|
return undefined
|
|
}
|
|
}
|
|
}
|
|
|
|
// Mock IntersectionObserver for tests
|
|
if (typeof globalThis.IntersectionObserver === 'undefined') {
|
|
globalThis.IntersectionObserver = class {
|
|
readonly root: Element | Document | null = null
|
|
readonly rootMargin: string = ''
|
|
readonly thresholds: ReadonlyArray<number> = []
|
|
constructor(_callback: IntersectionObserverCallback, _options?: IntersectionObserverInit) { /* noop */ }
|
|
observe() { /* noop */ }
|
|
unobserve() { /* noop */ }
|
|
disconnect() { /* noop */ }
|
|
takeRecords(): IntersectionObserverEntry[] { return [] }
|
|
}
|
|
}
|
|
|
|
// Mock Element.scrollIntoView for tests (not available in happy-dom/jsdom)
|
|
if (typeof Element !== 'undefined' && !Element.prototype.scrollIntoView)
|
|
Element.prototype.scrollIntoView = function () { /* noop */ }
|
|
|
|
afterEach(async () => {
|
|
// Wrap cleanup in act() to flush pending React scheduler work
|
|
// This prevents "window is not defined" errors from React 19's scheduler
|
|
// which uses setImmediate/MessageChannel that can fire after jsdom cleanup
|
|
await act(async () => {
|
|
cleanup()
|
|
})
|
|
})
|
|
|
|
// mock next/image to avoid width/height requirements for data URLs
|
|
vi.mock('next/image')
|
|
|
|
// mock foxact/use-clipboard - not available in test environment
|
|
vi.mock('foxact/use-clipboard', () => ({
|
|
useClipboard: () => ({
|
|
copy: vi.fn(),
|
|
copied: false,
|
|
}),
|
|
}))
|
|
|
|
// mock zustand - auto-resets all stores after each test
|
|
// Based on official Zustand testing guide: https://zustand.docs.pmnd.rs/guides/testing
|
|
vi.mock('zustand')
|
|
|
|
// mock react-i18next
|
|
vi.mock('react-i18next', async () => {
|
|
const actual = await vi.importActual<typeof import('react-i18next')>('react-i18next')
|
|
const { createReactI18nextMock } = await import('./test/i18n-mock')
|
|
return {
|
|
...actual,
|
|
...createReactI18nextMock(),
|
|
}
|
|
})
|
|
|
|
// mock window.matchMedia
|
|
Object.defineProperty(window, 'matchMedia', {
|
|
writable: true,
|
|
value: vi.fn().mockImplementation(query => ({
|
|
matches: false,
|
|
media: query,
|
|
onchange: null,
|
|
addListener: vi.fn(), // deprecated
|
|
removeListener: vi.fn(), // deprecated
|
|
addEventListener: vi.fn(),
|
|
removeEventListener: vi.fn(),
|
|
dispatchEvent: vi.fn(),
|
|
})),
|
|
})
|
|
|
|
// Mock localStorage for testing
|
|
const createMockLocalStorage = () => {
|
|
const storage: Record<string, string> = {}
|
|
return {
|
|
getItem: vi.fn((key: string) => storage[key] || null),
|
|
setItem: vi.fn((key: string, value: string) => {
|
|
storage[key] = value
|
|
}),
|
|
removeItem: vi.fn((key: string) => {
|
|
delete storage[key]
|
|
}),
|
|
clear: vi.fn(() => {
|
|
Object.keys(storage).forEach(key => delete storage[key])
|
|
}),
|
|
get storage() { return { ...storage } },
|
|
}
|
|
}
|
|
|
|
let mockLocalStorage: ReturnType<typeof createMockLocalStorage>
|
|
|
|
beforeEach(() => {
|
|
vi.clearAllMocks()
|
|
mockLocalStorage = createMockLocalStorage()
|
|
Object.defineProperty(globalThis, 'localStorage', {
|
|
value: mockLocalStorage,
|
|
writable: true,
|
|
configurable: true,
|
|
})
|
|
})
|