server
Nuxt 自动扫描这些目录中的文件,以注册支持热模块替换 (HMR) 的 API 和服务器处理程序。
-| server/
---| api/
-----| hello.ts # /api/hello
---| routes/
-----| bonjour.ts # /bonjour
---| middleware/
-----| log.ts # log all requests
每个文件都应该导出一个使用 defineEventHandler() 或 eventHandler() (别名) 定义的默认函数。
处理程序可以直接返回 JSON 数据、Promise,或者使用 event.node.res.end() 发送响应。
export default defineEventHandler((event) => {
return {
hello: 'world'
}
})
你现在可以在你的页面和组件中全局调用这个 API:
<script setup lang="ts">
const { data } = await useFetch('/api/hello')
</script>
<template>
<pre>{{ data }}</pre>
</template>
服务器路由
~/server/api 目录中的文件在其路由中自动添加 /api 前缀。
要添加没有 /api 前缀的服务器路由,请将它们放在 ~/server/routes 目录中。
示例:
export default defineEventHandler(() => 'Hello World!')
鉴于以上示例,/hello 路由将可以通过 http://localhost:3000/hello 访问。
服务器中间件
Nuxt 将自动读取 ~/server/middleware 中的任何文件,为你的项目创建服务器中间件。
中间件处理程序将在每个请求到达任何其他服务器路由之前运行,以添加或检查标头、记录请求或扩展事件的请求对象。
示例:
export default defineEventHandler((event) => {
console.log('New request: ' + getRequestURL(event))
})
export default defineEventHandler((event) => {
event.context.auth = { user: 123 }
})
服务器插件
Nuxt 将自动读取 ~/server/plugins 目录中的任何文件,并将它们注册为 Nitro 插件。这允许扩展 Nitro 的运行时行为并挂钩到生命周期事件。
示例:
export default defineNitroPlugin((nitroApp) => {
console.log('Nitro plugin', nitroApp)
})
Nuxt插件和 Nitro 插件之间的区别:
| 对比项 | Nuxt 插件 (plugins/) | Nitro 插件 (server/plugins/) |
|---|---|---|
| 属于谁? | 属于 Nuxt 应用 | 属于 Nitro(Nuxt 的服务端引擎) |
| 运行时机 | 应用初始化时执行(客户端或服务端) | 在服务器启动或每次请求时执行 |
| 文件位置 | plugins/ | server/plugins/ |
| 可用环境 | 客户端 & 服务端(视具体文件) | 只在服务端 |
| 常用于 | 用于扩展 Nuxt 应用上下文(注入全局方法、注册库、运行前处理逻辑) | 仅在 服务端运行(中间件、hook、日志、Header 注入、事件处理等) |
是否支持 useNuxtApp() | ✅ 是 | ❌ 否(使用 Nitro hooks) |
| 是否支持注入 | ✅ provide/inject | ❌ 不支持注入到 Nuxt app |
| 是否会执行在 API 请求中 | ✅(服务端插件) | ✅(每次请求都会执行) |
应用场景:
| 场景 | 用 Nuxt 插件 | 用 Nitro 插件 |
|---|---|---|
| 注册 Vue 插件(i18n、pinia 等) | ✅ 是 | ❌ 否 |
| 在组件中注入全局方法 | ✅ 是 | ❌ 否 |
| 处理请求/响应(如日志、header) | ❌ 否 | ✅ 是 |
| 运行服务端中间件逻辑 | ❌ 否 | ✅ 是 |
| 统一管理接口访问令牌 | ✅ 可与 useFetch 结合 | ✅ 更底层处理也可行 |
服务器工具
服务器路由由 unjs/h3 提供支持,它带有一组方便的辅助函数。
你可以在 ~/server/utils 目录中添加更多自己的辅助函数。
例如,你可以定义一个自定义的处理程序实用工具,该实用工具包装原始处理程序并在返回最终响应之前执行额外的操作。
示例:
import type { EventHandler, EventHandlerRequest } from 'h3'
export const defineWrappedResponseHandler = <T extends EventHandlerRequest, D> (
handler: EventHandler<T, D>
): EventHandler<T, D> =>
defineEventHandler<T>(async event => {
try {
// do something before the route handler
const response = await handler(event)
// do something after the route handler
return { response }
} catch (err) {
// Error handling
return { err }
}
})
服务器类型
为了提高 IDE 中 'nitro' 和 'vue' 自动导入之间的清晰度,你可以添加一个包含以下内容的 ~/server/tsconfig.json:
{
"extends": "../.nuxt/tsconfig.server.json"
}
目前,这些值在类型检查(nuxi typecheck)时不会被考虑,但你应该在 IDE 中获得更好的类型提示。
实践
路由参数
服务器路由可以在文件名中使用方括号中的动态参数,例如 /api/hello/[name].ts,并通过 event.context.params 访问。
export default defineEventHandler((event) => {
const name = getRouterParam(event, 'name')
return `Hello, ${name}!`
})
你现在可以在 /api/hello/nuxt 上全局调用此 API 并获得 Hello, nuxt!。
匹配 HTTP 方法
处理程序文件名可以带有 .get、.post、.put、.delete 等后缀,以匹配请求的 HTTP 方法。
export default defineEventHandler(() => 'Test get handler')
export default defineEventHandler(() => 'Test post handler')
鉴于以上示例,使用以下方法获取 /test:
- GET 方法:返回
Test get handler - POST 方法:返回
Test post handler - 任何其他方法:返回 405 错误
你也可以在目录中使用 index.[method].ts 来以不同的方式组织你的代码,这对于创建 API 命名空间很有用。
export default defineEventHandler((event) => {
// handle GET requests for the `api/foo` endpoint
})
export default defineEventHandler((event) => {
// handle POST requests for the `api/foo` endpoint
})
export default defineEventHandler((event) => {
// handle GET requests for the `api/foo/bar` endpoint
})
捕获所有路由
捕获所有路由对于处理回退路由很有用。
例如,创建一个名为 ~/server/api/foo/[...].ts 的文件将为所有与任何路由处理程序都不匹配的请求(例如 /api/foo/bar/baz)注册一个捕获所有路由。
export default defineEventHandler((event) => {
// event.context.path to get the route path: '/api/foo/bar/baz'
// event.context.params._ to get the route segment: 'bar/baz'
return `Default foo handler`
})
你可以通过使用 ~/server/api/foo/[...slug].ts 为捕获所有路由设置一个名称,并通过 event.context.params.slug 访问它。
export default defineEventHandler((event) => {
// event.context.params.slug to get the route segment: 'bar/baz'
return `Default foo handler`
})
Body 处理
export default defineEventHandler(async (event) => {
const body = await readBody(event)
return { body }
})
你现在可以使用以下方法全局调用此 API:
<script setup lang="ts">
async function submit() {
const { body } = await $fetch('/api/submit', {
method: 'post',
body: { test: 123 }
})
}
</script>
submit.post.ts 来匹配可以接受请求体的 POST 方法的请求。如果在 GET 请求中使用 readBody,readBody 将抛出 405 Method Not Allowed HTTP 错误。查询参数
示例查询 /api/query?foo=bar&baz=qux
export default defineEventHandler((event) => {
const query = getQuery(event)
return { a: query.foo, b: query.baz }
})
错误处理
如果没有抛出任何错误,将返回 200 OK 状态码。
任何未捕获的错误都将返回 500 Internal Server Error HTTP 错误。
要返回其他错误代码,请使用 createError 抛出异常:
export default defineEventHandler((event) => {
const id = parseInt(event.context.params.id) as number
if (!Number.isInteger(id)) {
throw createError({
statusCode: 400,
statusMessage: 'ID should be an integer',
})
}
return 'All good'
})
状态码
要返回其他状态码,请使用 setResponseStatus 实用程序。
例如,要返回 202 Accepted
export default defineEventHandler((event) => {
setResponseStatus(event, 202)
})
运行时配置
export default defineEventHandler(async (event) => {
const config = useRuntimeConfig(event)
const repo = await $fetch('https://api.github.com/repos/nuxt/nuxt', {
headers: {
Authorization: `token ${config.githubToken}`
}
})
return repo
})
export default defineNuxtConfig({
runtimeConfig: {
githubToken: ''
}
})
NUXT_GITHUB_TOKEN='<my-super-token>'
请求 Cookie
export default defineEventHandler((event) => {
const cookies = parseCookies(event)
return { cookies }
})
转发上下文和标头
默认情况下,在服务器路由中进行 fetch 请求时,既不会转发传入请求的标头,也不会转发请求上下文。你可以在服务器路由中使用 event.$fetch 来转发请求上下文和标头。
export default defineEventHandler((event) => {
return event.$fetch('/api/forwarded')
})
transfer-encoding、connection、keep-alive、upgrade、expect、host、accept响应后等待 Promise
在处理服务器请求时,你可能需要执行不应阻塞客户端响应的异步任务(例如,缓存和日志记录)。你可以使用 event.waitUntil 在后台等待 Promise,而不会延迟响应。
event.waitUntil 方法接受一个 Promise,该 Promise 将在处理程序终止之前等待,确保即使服务器在发送响应后立即终止处理程序,任务也会完成。这与运行时提供程序集成,以利用其处理发送响应后异步操作的本机功能。
const timeConsumingBackgroundTask = async () => {
await new Promise((resolve) => setTimeout(resolve, 1000))
};
export default eventHandler((event) => {
// schedule a background task without blocking the response
event.waitUntil(timeConsumingBackgroundTask())
// immediately send the response to the client
return 'done'
});
高级用法
Nitro 配置
你可以在 nuxt.config 中使用 nitro 键直接设置 Nitro 配置。
export default defineNuxtConfig({
// https://nitro.unjs.io/config
nitro: {}
})
嵌套路由
import { createRouter, defineEventHandler, useBase } from 'h3'
const router = createRouter()
router.get('/test', defineEventHandler(() => 'Hello World'))
export default useBase('/api/hello', router.handler)
发送流
import fs from 'node:fs'
import { sendStream } from 'h3'
export default defineEventHandler((event) => {
return sendStream(event, fs.createReadStream('/path/to/file'))
})
发送重定向
export default defineEventHandler(async (event) => {
await sendRedirect(event, '/path/redirect/to', 302)
})
遗留处理程序或中间件
export default fromNodeMiddleware((req, res) => {
res.end('Legacy handler')
})
可以使用 unjs/h3 实现遗留支持,但建议尽可能避免使用遗留处理程序。
export default fromNodeMiddleware((req, res, next) => {
console.log('Legacy middleware')
next()
})
next() 回调与 async 或返回 Promise 的遗留中间件组合使用。服务器存储
Nitro 提供了一个跨平台的 存储层。为了配置额外的存储挂载点,你可以使用 nitro.storage 或 服务器插件。
添加 Redis 存储的示例:
使用 nitro.storage:
export default defineNuxtConfig({
nitro: {
storage: {
redis: {
driver: 'redis',
/* redis connector options */
port: 6379, // Redis port
host: "127.0.0.1", // Redis host
username: "", // needs Redis >= 6
password: "",
db: 0, // Defaults to 0
tls: {} // tls/ssl
}
}
}
})
然后在你的 API 处理程序中:
export default defineEventHandler(async (event) => {
// List all keys with
const keys = await useStorage('redis').getKeys()
// Set a key with
await useStorage('redis').setItem('foo', 'bar')
// Remove a key with
await useStorage('redis').removeItem('foo')
return {}
})
或者,你可以使用服务器插件和运行时配置创建一个存储挂载点:
import redisDriver from 'unstorage/drivers/redis'
export default defineNitroPlugin(() => {
const storage = useStorage()
// Dynamically pass in credentials from runtime configuration, or other sources
const driver = redisDriver({
base: 'redis',
host: useRuntimeConfig().redis.host,
port: useRuntimeConfig().redis.port,
/* other redis connector options */
})
// Mount driver
storage.mount('redis', driver)
})
export default defineNuxtConfig({
runtimeConfig: {
redis: { // Default values
host: '',
port: 0,
/* other redis connector options */
}
}
})