Skip to content

Commit

Permalink
fix: preview.allowedHosts with specific values was not respected (#…
Browse files Browse the repository at this point in the history
  • Loading branch information
sapphi-red committed Jan 21, 2025
1 parent a1824c5 commit 9df6e6b
Show file tree
Hide file tree
Showing 4 changed files with 26 additions and 14 deletions.
2 changes: 1 addition & 1 deletion packages/vite/src/node/preview.ts
Original file line number Diff line number Diff line change
Expand Up @@ -194,7 +194,7 @@ export async function preview(
const { allowedHosts } = config.preview
// no need to check for HTTPS as HTTPS is not vulnerable to DNS rebinding attacks
if (allowedHosts !== true && !config.preview.https) {
app.use(hostCheckMiddleware(config))
app.use(hostCheckMiddleware(config, true))
}

// proxy
Expand Down
2 changes: 1 addition & 1 deletion packages/vite/src/node/server/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -858,7 +858,7 @@ export async function _createServer(
const { allowedHosts } = serverConfig
// no need to check for HTTPS as HTTPS is not vulnerable to DNS rebinding attacks
if (allowedHosts !== true && !serverConfig.https) {
middlewares.use(hostCheckMiddleware(config))
middlewares.use(hostCheckMiddleware(config, false))
}

middlewares.use(cachedTransformMiddleware(server))
Expand Down
34 changes: 23 additions & 11 deletions packages/vite/src/node/server/middlewares/hostCheck.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,8 @@ import type { Connect } from 'dep-types/connect'
import type { ResolvedConfig } from '../../config'
import type { ResolvedPreviewOptions, ResolvedServerOptions } from '../..'

const allowedHostsCache = new WeakMap<ResolvedConfig, Set<string>>()
const allowedHostsServerCache = new WeakMap<ResolvedConfig, Set<string>>()
const allowedHostsPreviewCache = new WeakMap<ResolvedConfig, Set<string>>()

const isFileOrExtensionProtocolRE = /^(?:file|.+-extension):/i

Expand Down Expand Up @@ -118,48 +119,59 @@ export function isHostAllowedWithoutCache(

/**
* @param config resolved config
* @param isPreview whether it's for the preview server or not
* @param host the value of host header. See [RFC 9110 7.2](https://datatracker.ietf.org/doc/html/rfc9110#name-host-and-authority).
*/
export function isHostAllowed(config: ResolvedConfig, host: string): boolean {
if (config.server.allowedHosts === true) {
export function isHostAllowed(
config: ResolvedConfig,
isPreview: boolean,
host: string,
): boolean {
const allowedHosts = isPreview
? config.preview.allowedHosts
: config.server.allowedHosts
if (allowedHosts === true) {
return true
}

if (!allowedHostsCache.has(config)) {
allowedHostsCache.set(config, new Set())
const cache = isPreview ? allowedHostsPreviewCache : allowedHostsServerCache
if (!cache.has(config)) {
cache.set(config, new Set())
}

const allowedHosts = allowedHostsCache.get(config)!
if (allowedHosts.has(host)) {
const cachedAllowedHosts = cache.get(config)!
if (cachedAllowedHosts.has(host)) {
return true
}

const result = isHostAllowedWithoutCache(
config.server.allowedHosts ?? [],
allowedHosts ?? [],
config.additionalAllowedHosts,
host,
)
if (result) {
allowedHosts.add(host)
cachedAllowedHosts.add(host)
}
return result
}

export function hostCheckMiddleware(
config: ResolvedConfig,
isPreview: boolean,
): Connect.NextHandleFunction {
// Keep the named function. The name is visible in debug logs via `DEBUG=connect:dispatcher ...`
return function viteHostCheckMiddleware(req, res, next) {
const hostHeader = req.headers.host
if (!hostHeader || !isHostAllowed(config, hostHeader)) {
if (!hostHeader || !isHostAllowed(config, isPreview, hostHeader)) {
const hostname = hostHeader?.replace(/:\d+$/, '')
const hostnameWithQuotes = JSON.stringify(hostname)
const optionName = `${isPreview ? 'preview' : 'server'}.allowedHosts`
res.writeHead(403, {
'Content-Type': 'text/plain',
})
res.end(
`Blocked request. This host (${hostnameWithQuotes}) is not allowed.\n` +
`To allow this host, add ${hostnameWithQuotes} to \`server.allowedHosts\` in vite.config.js.`,
`To allow this host, add ${hostnameWithQuotes} to \`${optionName}\` in vite.config.js.`,
)
return
}
Expand Down
2 changes: 1 addition & 1 deletion packages/vite/src/node/server/ws.ts
Original file line number Diff line number Diff line change
Expand Up @@ -155,7 +155,7 @@ export function createWebSocketServer(

const shouldHandle = (req: IncomingMessage) => {
const hostHeader = req.headers.host
if (!hostHeader || !isHostAllowed(config, hostHeader)) {
if (!hostHeader || !isHostAllowed(config, false, hostHeader)) {
return false
}

Expand Down

0 comments on commit 9df6e6b

Please sign in to comment.