import { Script, createContext } from 'node:vm' import { expect } from 'vitest' import type { Page } from 'playwright-core' import { parse } from 'devalue' import { reactive, ref, shallowReactive, shallowRef } from 'vue' import { createError } from 'h3' import { getBrowser, url, useTestContext } from '@nuxt/test-utils/e2e' export const isRenderingJson = process.env.TEST_PAYLOAD !== 'js' export async function renderPage (path = '/') { const ctx = useTestContext() if (!ctx.options.browser) { throw new Error('`renderPage` require `options.browser` to be set') } const browser = await getBrowser() const page = await browser.newPage({}) const pageErrors: Error[] = [] const requests: string[] = [] const consoleLogs: { type: string, text: string }[] = [] page.on('console', (message) => { consoleLogs.push({ type: message.type(), text: message.text(), }) }) page.on('pageerror', (err) => { pageErrors.push(err) }) page.on('request', (req) => { try { requests.push(req.url().replace(url('/'), '/')) } catch { // TODO } }) if (path) { await gotoPath(page, path) } return { page, pageErrors, requests, consoleLogs, } } export async function expectNoClientErrors (path: string) { const ctx = useTestContext() if (!ctx.options.browser) { return } const { page, pageErrors, consoleLogs } = (await renderPage(path))! expect(pageErrors).toEqual([]) expectNoErrorsOrWarnings(consoleLogs) await page.close() } export function expectNoErrorsOrWarnings (consoleLogs: Array<{ type: string, text: string }>) { const consoleLogErrors = consoleLogs.filter(i => i.type === 'error') const consoleLogWarnings = consoleLogs.filter(i => i.type === 'warning') expect(consoleLogErrors).toEqual([]) expect(consoleLogWarnings).toEqual([]) } export async function gotoPath (page: Page, path: string) { await page.goto(url(path)) await page.waitForFunction(path => window.useNuxtApp?.()._route.fullPath === path && !window.useNuxtApp?.().isHydrating, path) } type EqualityVal = string | number | boolean | null | undefined | RegExp export async function expectWithPolling ( get: () => Promise | EqualityVal, expected: EqualityVal, retries = process.env.CI ? 100 : 30, delay = process.env.CI ? 500 : 100, ) { let result: EqualityVal for (let i = retries; i >= 0; i--) { result = await get() if (result?.toString() === expected?.toString()) { break } await new Promise(resolve => setTimeout(resolve, delay)) } expect(result?.toString(), `"${result?.toString()}" did not equal "${expected?.toString()}" in ${retries * delay}ms`).toEqual(expected?.toString()) } const revivers = { NuxtError: (data: any) => createError(data), EmptyShallowRef: (data: any) => shallowRef(JSON.parse(data)), EmptyRef: (data: any) => ref(JSON.parse(data)), ShallowRef: (data: any) => shallowRef(data), ShallowReactive: (data: any) => shallowReactive(data), Island: (key: any) => key, Ref: (data: any) => ref(data), Reactive: (data: any) => reactive(data), // test fixture reviver only BlinkingText: () => '', } export function parsePayload (payload: string) { return parse(payload || '', revivers) } export function parseData (html: string) { if (!isRenderingJson) { const { script = '' } = html.match(/