在 Next.js App Router 中使用仅在服务端的代码

React Server Components 架构将组件分为客户端和服务端类型,并与 Next.js 的 App Router 集成。

使用 App Router 进行开发时,区分服务端代码和客户端代码对于应用程序的安全性、性能和可维护性至关重要。这篇博文将介绍在 Next.js 应用程序中定义服务端代码。

服务器端代码

Next.js 中的 Server-only code 是指仅在服务器端执行的代码。这可能包括以下模块或功能:

  • 使用特定于服务器的库
  • 访问包含敏感信息的环境变量
  • 与数据库或外部 API 交互
  • 处理机密业务逻辑

之所以会出现这样的挑战,是因为 JavaScript 模块可以在服务器和客户端组件之间共享,从而导致服务器端代码无意中包含在客户端包中。这种包含可能会暴露敏感数据、增加捆绑包大小并导致潜在的安全漏洞。

实践示例

让我们通过一个实践示例来演示服务端代码。首先使用 create-next-app 创建 Next.js 应用程序。然后,在你创建的 src/utils 目录中,添加一个名为 server-utils.ts 的文件。该文件将包含以下服务端函数:

js 复制代码
// src/utils/server-utils.ts
export const serverSideFunction = () => {
  console.log(
    `Using multiple libraries,
     accessing environment variables,
     interacting with a database,
     processing confidential information`
  );
  return "Server-side result";
};

想象一下这个函数使用各种 NPM 包、访问 API 密钥、从数据库检索数据并处理敏感算法。这样的函数永远不应该暴露给客户端。

现在,让我们在服务器组件中使用此函数,例如"关于"页面组件:

js 复制代码
// pages/about.tsx
import { serverSideFunction } from "@/utils/server-utils";

const About = () => {
  const result = serverSideFunction();
  console.log(result);
  return <div>About Page</div>;
};

export default About;

导航到 /about 路径,你将在服务器终端(而不是浏览器)中观察到日志消息,这表明我们的代码仅适用于服务器。

但是,如果此函数被错误地导入到客户端组件(例如仪表板页面)中怎么办?该代码可能部分或完全不起作用。这就是 server-only 包发挥作用的地方。

使用 server-only

为了保护我们的应用程序,我们安装 server-only 包:

shell 复制代码
npm i server-only

然后我们修改 server-utils.ts 以包含此包:

js 复制代码
// src/utils/server-utils.ts
import 'server-only';

export const serverSideFunction = () => {
  console.log(
    `Using multiple libraries,
     accessing environment variables,
     interacting with a database,
     processing confidential information`
  );
  return "Server-side result";
};

现在,如果有人尝试将 serverSideFunction 导入客户端组件,构建过程将抛出错误,从而防止服务器端代码泄漏到客户端。

使用 client-only

正如服务端代码需要隔离一样,与特定于浏览器的功能(如 DOM、窗口对象、localStorage 等)交互的仅客户端代码也必须限制在客户端,以有效地利用特定于浏览器的功能。

client-only 包充当护栏,确保我们的客户端代码保留在其所属的位置。

总结

在 Next.js 应用程序中,保持服务端代码和客户端代码之间的清晰界限至关重要。它确保你的应用程序的完整性、安全性和用户体验。使用 server-onlyclient-only 包并遵循最佳实践来加强这种分离并保护你的应用程序。

相关推荐
ZC跨境爬虫1 小时前
跟着 MDN 学CSS day_39:(Flexbox 弹性盒子核心机制)
前端·css·ui·html·tensorflow
小陈同学呦2 小时前
前端如何处理订单状态导航的数据竞态问题
前端·javascript
开发者每周简报2 小时前
网海三部曲·无名宗师传
javascript·人工智能
喵个咪2 小时前
GoWind Toolkit 前端代码生成|Vue3(ElementPlus/Vben)、React(AntDesign)全自动一键生成教程
前端·vue.js·react.js
摆烂大大王3 小时前
玩转 OpenClaw:用 TaskFlow + Heartbeat 打造自动化工作流
前端·人工智能·自动化
zhangxingchao3 小时前
AI 大模型核心六:量化、Workflow 与 Agent、多轮 RAG
前端·人工智能·后端
梦想的颜色4 小时前
TypeScript 完全指南(上):从零开始掌握类型系统
前端·typescript
之歆4 小时前
Day01_ES6+ 专业指南:从基础到实战的现代JavaScript开发(下)
前端·javascript·es6
花椒技术4 小时前
复杂直播业务做 RN 跨端,我们最后保留了哪些 Native 边界
react native·react.js·harmonyos
lichenyang4534 小时前
鸿蒙 MVVM 实战:从 Demo 到工程化,聊聊登录、状态管理与埋点系统设计
前端