Skip to content
Permalink

Comparing changes

Choose two branches to see what’s changed or to start a new pull request. If you need to, you can also or learn more about diff comparisons.

Open a pull request

Create a new pull request by comparing changes across two branches. If you need to, you can also . Learn more about diff comparisons here.
base repository: vitejs/vite
Failed to load repositories. Confirm that selected base ref is valid, then try again.
Loading
base: v6.2.3
Choose a base ref
...
head repository: vitejs/vite
Failed to load repositories. Confirm that selected head ref is valid, then try again.
Loading
compare: v6.2.4
Choose a head ref
  • 2 commits
  • 5 files changed
  • 2 contributors

Commits on Mar 31, 2025

  1. fix: fs check in transform middleware (#19761)

    patak-dev committed Mar 31, 2025
    Copy the full SHA
    7a4faba View commit details
  2. release: v6.2.4

    patak-dev committed Mar 31, 2025
    Copy the full SHA
    037f801 View commit details
6 changes: 6 additions & 0 deletions packages/vite/CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,9 @@
## <small>6.2.4 (2025-03-31)</small>

* fix: fs check in transform middleware (#19761) ([7a4faba](https://github.com/vitejs/vite/commit/7a4fabab6a3aa24c89144e15a13d78f92b52e588)), closes [#19761](https://github.com/vitejs/vite/issues/19761)



## <small>6.2.3 (2025-03-24)</small>

* fix: fs raw query with query separators (#19702) ([f234b57](https://github.com/vitejs/vite/commit/f234b5744d8b74c95535a7b82cc88ed2144263c1)), closes [#19702](https://github.com/vitejs/vite/issues/19702)
2 changes: 1 addition & 1 deletion packages/vite/package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "vite",
"version": "6.2.3",
"version": "6.2.4",
"type": "module",
"license": "MIT",
"author": "Evan You",
10 changes: 7 additions & 3 deletions packages/vite/src/node/server/middlewares/transform.ts
Original file line number Diff line number Diff line change
@@ -12,10 +12,8 @@ import {
isJSRequest,
normalizePath,
prettifyUrl,
rawRE,
removeImportQuery,
removeTimestampQuery,
urlRE,
} from '../../utils'
import { send } from '../send'
import { ERR_LOAD_URL, transformRequest } from '../transformRequest'
@@ -45,6 +43,11 @@ const debugCache = createDebugger('vite:cache')
const knownIgnoreList = new Set(['/', '/favicon.ico'])
const trailingQuerySeparatorsRE = /[?&]+$/

// TODO: consolidate this regex pattern with the url, raw, and inline checks in plugins
const urlRE = /[?&]url\b/
const rawRE = /[?&]raw\b/
const inlineRE = /[?&]inline\b/

/**
* A middleware that short-circuits the middleware chain to serve cached transformed modules
*/
@@ -176,7 +179,8 @@ export function transformMiddleware(
)
if (
(rawRE.test(urlWithoutTrailingQuerySeparators) ||
urlRE.test(urlWithoutTrailingQuerySeparators)) &&
urlRE.test(urlWithoutTrailingQuerySeparators) ||
inlineRE.test(urlWithoutTrailingQuerySeparators)) &&
!ensureServingAccess(
urlWithoutTrailingQuerySeparators,
server,
24 changes: 24 additions & 0 deletions playground/fs-serve/__tests__/fs-serve.spec.ts
Original file line number Diff line number Diff line change
@@ -67,6 +67,18 @@ describe.runIf(isServe)('main', () => {
expect(await page.textContent('.unsafe-fetch-8498-2-status')).toBe('404')
})

test('unsafe fetch import inline', async () => {
expect(await page.textContent('.unsafe-fetch-import-inline-status')).toBe(
'403',
)
})

test('unsafe fetch raw query import', async () => {
expect(
await page.textContent('.unsafe-fetch-raw-query-import-status'),
).toBe('403')
})

test('safe fs fetch', async () => {
expect(await page.textContent('.safe-fs-fetch')).toBe(stringified)
expect(await page.textContent('.safe-fs-fetch-status')).toBe('200')
@@ -120,6 +132,18 @@ describe.runIf(isServe)('main', () => {
expect(await page.textContent('.unsafe-fs-fetch-8498-2-status')).toBe('404')
})

test('unsafe fs fetch import inline', async () => {
expect(
await page.textContent('.unsafe-fs-fetch-import-inline-status'),
).toBe('403')
})

test('unsafe fs fetch import inline wasm init', async () => {
expect(
await page.textContent('.unsafe-fs-fetch-import-inline-wasm-init-status'),
).toBe('403')
})

test('nested entry', async () => {
expect(await page.textContent('.nested-entry')).toBe('foobar')
})
51 changes: 51 additions & 0 deletions playground/fs-serve/root/src/index.html
Original file line number Diff line number Diff line change
@@ -23,6 +23,8 @@ <h2>Unsafe Fetch</h2>
<pre class="unsafe-fetch-8498"></pre>
<pre class="unsafe-fetch-8498-2-status"></pre>
<pre class="unsafe-fetch-8498-2"></pre>
<pre class="unsafe-fetch-import-inline-status"></pre>
<pre class="unsafe-fetch-raw-query-import-status"></pre>

<h2>Safe /@fs/ Fetch</h2>
<pre class="safe-fs-fetch-status"></pre>
@@ -45,6 +47,8 @@ <h2>Unsafe /@fs/ Fetch</h2>
<pre class="unsafe-fs-fetch-8498"></pre>
<pre class="unsafe-fs-fetch-8498-2-status"></pre>
<pre class="unsafe-fs-fetch-8498-2"></pre>
<pre class="unsafe-fs-fetch-import-inline-status"></pre>
<pre class="unsafe-fs-fetch-import-inline-wasm-init-status"></pre>

<h2>Nested Entry</h2>
<pre class="nested-entry"></pre>
@@ -160,6 +164,24 @@ <h2>Denied</h2>
console.error(e)
})

// outside of allowed dir with import inline
fetch(joinUrlSegments(base, '/unsafe.txt?import&inline'))
.then((r) => {
text('.unsafe-fetch-import-inline-status', r.status)
})
.catch((e) => {
console.error(e)
})

// outside of allowed dir with raw query import
fetch(joinUrlSegments(base, '/unsafe.txt?raw?import'))
.then((r) => {
text('.unsafe-fetch-raw-query-import-status', r.status)
})
.catch((e) => {
console.error(e)
})

// imported before, should be treated as safe
fetch(joinUrlSegments(base, joinUrlSegments('/@fs/', ROOT) + '/safe.json'))
.then((r) => {
@@ -247,6 +269,35 @@ <h2>Denied</h2>
console.error(e)
})

// outside of root inline
fetch(
joinUrlSegments(
base,
joinUrlSegments('/@fs/', ROOT) + '/root/unsafe.txt?import&inline',
),
)
.then((r) => {
text('.unsafe-fs-fetch-import-inline-status', r.status)
})
.catch((e) => {
console.error(e)
})

// outside of root inline, faux wasm?init
fetch(
joinUrlSegments(
base,
joinUrlSegments('/@fs/', ROOT) +
'/root/unsafe.txt?import&?inline=1.wasm?init',
),
)
.then((r) => {
text('.unsafe-fs-fetch-import-inline-wasm-init-status', r.status)
})
.catch((e) => {
console.error(e)
})

// outside root with special characters #8498
fetch(
joinUrlSegments(