Some checks failed
Test examples / Test Examples (20) (push) Has been cancelled
Test examples / Test Examples (22) (push) Has been cancelled
Lock Threads / action (push) Has been cancelled
Trigger Release / start (push) Has been cancelled
Stale issue handler / stale (push) Has been cancelled
Update Font Data / create-pull-request (push) Has been cancelled
build-and-deploy / deploy-target (push) Has been cancelled
build-and-deploy / build (push) Has been cancelled
build-and-deploy / stable - aarch64-unknown-linux-musl - node@16 (push) Has been cancelled
build-and-deploy / stable - x86_64-unknown-linux-musl - node@16 (push) Has been cancelled
build-and-deploy / stable - aarch64-unknown-linux-gnu - node@16 (push) Has been cancelled
build-and-deploy / stable - x86_64-unknown-linux-gnu - node@16 (push) Has been cancelled
build-and-deploy / stable - aarch64-pc-windows-msvc - node@16 (push) Has been cancelled
build-and-deploy / stable - x86_64-pc-windows-msvc - node@16 (push) Has been cancelled
build-and-deploy / stable - aarch64-apple-darwin - node@16 (push) Has been cancelled
build-and-deploy / stable - x86_64-apple-darwin - node@16 (push) Has been cancelled
build-and-deploy / build-wasm (nodejs) (push) Has been cancelled
build-and-deploy / build-wasm (web) (push) Has been cancelled
build-and-deploy / Deploy preview tarball (push) Has been cancelled
build-and-deploy / Potentially publish release (push) Has been cancelled
build-and-deploy / publish-turbopack-npm-packages (push) Has been cancelled
build-and-deploy / Deploy examples (push) Has been cancelled
build-and-deploy / thank you, build (push) Has been cancelled
build-and-deploy / Upload Turbopack Bytesize metrics to Datadog (push) Has been cancelled
Rspack Next.js development integration tests / Rspack integration tests (push) Has been cancelled
Rspack Next.js production integration tests / Rspack integration tests (push) Has been cancelled
Turbopack Next.js development integration tests / Next.js integration tests (push) Has been cancelled
Turbopack Next.js production integration tests / Next.js integration tests (push) Has been cancelled
Update Rspack test manifest / Update and upload Rspack development test manifest (push) Has been cancelled
Update Rspack test manifest / Update and upload Rspack production test manifest (push) Has been cancelled
Upload bundler test manifests to areweturboyet.com / Upload test results (push) Has been cancelled
Update React / create-pull-request (push) Has been cancelled
test-e2e-project-reset-cron / reset-test-project (push) Has been cancelled
Notify about the top 15 issues/PRs/feature requests (most reacted) in the last 90 days / run (push) Has been cancelled
220 lines
6.8 KiB
TypeScript
220 lines
6.8 KiB
TypeScript
/* eslint-env jest */
|
|
|
|
import {
|
|
waitForRedbox,
|
|
findPort,
|
|
getRedboxHeader,
|
|
killApp,
|
|
launchApp,
|
|
nextBuild,
|
|
nextStart,
|
|
waitFor,
|
|
check,
|
|
fetchViaHTTP,
|
|
} from 'next-test-utils'
|
|
import cheerio from 'cheerio'
|
|
import webdriver from 'next-webdriver'
|
|
import { join } from 'path'
|
|
|
|
let app: Awaited<ReturnType<typeof nextStart>>
|
|
let appPort: number
|
|
const appDir = join(__dirname, '..')
|
|
|
|
// This test doesn't seem to benefit from retries, let's disable them until the test gets fixed
|
|
// to prevent long running times
|
|
jest.retryTimes(0)
|
|
|
|
const showsError = async (pathname, regex, click = false, isWarn = false) => {
|
|
const browser = await webdriver(appPort, pathname)
|
|
try {
|
|
// wait for page to be built and navigated to
|
|
await browser.waitForElementByCss('#click-me')
|
|
if (isWarn) {
|
|
await browser.eval(`(function() {
|
|
window.warnLogs = []
|
|
var origWarn = window.console.warn
|
|
window.console.warn = (...args) => {
|
|
window.warnLogs.push(args.join(' '))
|
|
origWarn.apply(window.console, args)
|
|
}
|
|
})()`)
|
|
}
|
|
if (click) {
|
|
await browser.elementByCss('#click-me').click()
|
|
await waitFor(500)
|
|
}
|
|
if (isWarn) {
|
|
await check(async () => {
|
|
const warnLogs = await browser.eval('window.warnLogs')
|
|
return warnLogs.join('\n')
|
|
}, regex)
|
|
} else {
|
|
await waitForRedbox(browser)
|
|
const errorContent = await getRedboxHeader(browser)
|
|
expect(errorContent).toMatch(regex)
|
|
}
|
|
} finally {
|
|
await browser.close()
|
|
}
|
|
}
|
|
|
|
const noError = async (pathname, click = false) => {
|
|
const browser = await webdriver(appPort, '/')
|
|
try {
|
|
await browser.eval(`(function() {
|
|
window.caughtErrors = []
|
|
window.addEventListener('error', function (error) {
|
|
window.caughtErrors.push(error.message || 1)
|
|
})
|
|
window.addEventListener('unhandledrejection', function (error) {
|
|
window.caughtErrors.push(error.message || 1)
|
|
})
|
|
window.next.router.replace('${pathname}')
|
|
})()`)
|
|
await browser.waitForElementByCss('#click-me')
|
|
if (click) {
|
|
await browser.elementByCss('#click-me').click()
|
|
await waitFor(500)
|
|
}
|
|
const caughtErrors = await browser.eval(`window.caughtErrors`)
|
|
expect(caughtErrors).toHaveLength(0)
|
|
} finally {
|
|
await browser.close()
|
|
}
|
|
}
|
|
|
|
describe('Invalid hrefs', () => {
|
|
;(process.env.TURBOPACK_DEV ? describe.skip : describe)(
|
|
'production mode',
|
|
() => {
|
|
beforeAll(async () => {
|
|
await nextBuild(appDir)
|
|
appPort = await findPort()
|
|
app = await nextStart(appDir, appPort)
|
|
})
|
|
afterAll(() => killApp(app))
|
|
|
|
it('does not show error in production when mailto: is used as href on Link', async () => {
|
|
await noError('/first')
|
|
})
|
|
|
|
it('does not show error in production when https:// is used in href on Link', async () => {
|
|
await noError('/second')
|
|
})
|
|
|
|
it('does not show error in production when exotic protocols are used in href in Link', async () => {
|
|
const browser = await webdriver(appPort, '/exotic-href')
|
|
|
|
expect(
|
|
(await browser.log()).filter((x) => x.source === 'error')
|
|
).toEqual([])
|
|
})
|
|
|
|
it('does not show error when internal href is used with external as', async () => {
|
|
await noError('/invalid-relative', true)
|
|
})
|
|
|
|
it('shows error when dynamic route mismatch is used on Link', async () => {
|
|
const browser = await webdriver(appPort, '/dynamic-route-mismatch')
|
|
try {
|
|
await browser.eval(`(function() {
|
|
window.caughtErrors = []
|
|
window.addEventListener('unhandledrejection', (error) => {
|
|
window.caughtErrors.push(error.reason.message)
|
|
})
|
|
})()`)
|
|
await browser.elementByCss('a').click()
|
|
await waitFor(500)
|
|
const errors = await browser.eval('window.caughtErrors')
|
|
expect(
|
|
errors.find((err) =>
|
|
err.includes(
|
|
'The provided `as` value (/blog/post-1) is incompatible with the `href` value (/[post]). Read more: https://nextjs.org/docs/messages/incompatible-href-as'
|
|
)
|
|
)
|
|
).toBeTruthy()
|
|
} finally {
|
|
await browser.close()
|
|
}
|
|
})
|
|
|
|
it("doesn't fail on invalid url", async () => {
|
|
await noError('/third')
|
|
})
|
|
|
|
it('renders a link with invalid href', async () => {
|
|
const res = await fetchViaHTTP(appPort, '/third')
|
|
const $ = cheerio.load(await res.text())
|
|
expect($('#click-me').attr('href')).toBe('https://')
|
|
})
|
|
|
|
it('renders a link with mailto: href', async () => {
|
|
const res = await fetchViaHTTP(appPort, '/first')
|
|
const $ = cheerio.load(await res.text())
|
|
expect($('#click-me').attr('href')).toBe('mailto:idk@idk.com')
|
|
})
|
|
}
|
|
)
|
|
;(process.env.TURBOPACK_BUILD ? describe.skip : describe)(
|
|
'development mode',
|
|
() => {
|
|
beforeAll(async () => {
|
|
appPort = await findPort()
|
|
app = await launchApp(appDir, appPort)
|
|
})
|
|
afterAll(() => killApp(app))
|
|
|
|
it('does not show error when mailto: is used as href on Link', async () => {
|
|
await noError('/first')
|
|
})
|
|
|
|
it('does not show error when https:// is used as href in Link', async () => {
|
|
await noError('/second')
|
|
})
|
|
|
|
it('does not show error when exotic protocols are used in href in Link', async () => {
|
|
const browser = await webdriver(appPort, '/exotic-href')
|
|
|
|
expect(
|
|
(await browser.log()).filter((x) => x.source === 'error')
|
|
).toEqual([])
|
|
})
|
|
|
|
// eslint-disable-next-line jest/no-identical-title
|
|
it('shows error when dynamic route mismatch is used on Link', async () => {
|
|
await showsError(
|
|
'/dynamic-route-mismatch',
|
|
/The provided `as` value \(\/blog\/post-1\) is incompatible with the `href` value \(\/\[post\]\)/,
|
|
true
|
|
)
|
|
})
|
|
|
|
it('shows error when internal href is used with external as', async () => {
|
|
await showsError(
|
|
'/invalid-relative',
|
|
/Invalid href: "\/second" and as: "mailto:hello@example\.com", received relative href and external as/,
|
|
true
|
|
)
|
|
})
|
|
|
|
it('does not throw error when dynamic route mismatch is used on Link and params are manually provided', async () => {
|
|
await noError('/dynamic-route-mismatch-manual', true)
|
|
})
|
|
|
|
// eslint-disable-next-line jest/no-identical-title
|
|
it("doesn't fail on invalid url", async () => {
|
|
await noError('/third')
|
|
})
|
|
|
|
it('shows warning when dynamic route mismatch is used on Link', async () => {
|
|
await showsError(
|
|
'/dynamic-route-mismatch',
|
|
/Mismatching `as` and `href` failed to manually provide the params: post in the `href`'s `query`/,
|
|
true,
|
|
true
|
|
)
|
|
})
|
|
}
|
|
)
|
|
})
|