构建服务端渲染项目
其实在前一节中,大部分的主要代码都已经通过示例列出来了,在这一节里再重新完整整理一遍。
本书中服务端渲染的示例都采用Bun运行时 + Elysia框架的高性能服务组织形式,这套组合与网络上常见的NodeJS运行时 + Express框架的服务组织形式有所不同,例如NodeJS运行时是不支持ReadableStream
的,所以在server.ts
中就需要使用renderToPipeableStream
来获取NodeJS运行时支持的PipeableStream
输出渲染完成的HTML片段。
另外Bun运行时是原生提供了Typescript的支持的,所以书中的Typescript文件可以直接运行,但是如果使用NodeJS运行时就需要使用Babel等工具进行实时转译或者在编译后执行。
传统 React 项目的构建
构建一个 React SSR 项目的基础还是一个传统的 React 项目,这里采用 Vite 构建这个 React 项目。对于本节的示例来说,可以使用以下命令来创建。
bun create vite react-ssr-example --template react-ts
创建项目使用的模板还可以使用react-swc-ts
以提高React项目的编译性能,亦或者不指定模板,在选择模板的时候选择Others > create-vite-extra
来直接创建一个直接支持SSR的项目。
构建好的项目中一般会存在index.html
文件,在src
目录下会存在main.tsx
文件和App.tsx
文件。
接下来,就需要对这个项目做一些改造,这个改造主要按照以下思路进行:
- 将
src/App.tsx
文件改造成 SSR 的渲染入口,也就是包含启动 HTML 的文件。 - 将
src/main.tsx
文件改造成通用 React 组件树入口。 - 增加一个
server.ts
文件用于应用服务器的启动文件。
改造 SSR 渲染入口
这里还是要使用前一节中所说到的全部文档托管的渲染入口组织形式。所以App.tsx
文件的内容如下所示。
import { StrictMode } from "react";
import { Route, Routes, StaticRouter } from "react-router-dom";
import { MainLayout } from "./pages/MainLayout";
export const App = () => {
return (
<StrictMode>
<html lang="zh-CN">
<head>
<meta charSet="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<title>React SSR App</title>
<link rel="stylesheet" href="/src/index.css" />
</head>
<body>
<StaticRouter>
<Routes>
<Route path="/" element={<MainLayout />} />
</Routes>
</StaticRouter>
</body>
</html>
</StrictMode>
);
};
在网络上其他的示例中,这个 SSR 渲染入口文件也常常被命名为entry-server.js
、entry-server.jsx
、entry-server.tsx
等。但是无论其如何命名,所执行的功能都是声明项目最基础的 HTML 文件和载入 React 应用入口文件。
在定义了App.tsx
文件以后,就可以将项目根目录中的index.html
文件删除了。
在 React 19 中,HTML 的元标签,比如<html>
、<meta>
、<title>
等标签都是可以在组件中使用了,而且会被自动放置到 HTML DOM 中的相应位置。
与上一节中部分托管的形式不同,这一份App.tsx
文件中虽然也不需要提供通用 React 组件树入口文件的载入,但是还是需要提供所使用的样式文件的引入。这个样式文件的引入直接仿照示例中的<link>
标签的书写即可。
React 应用的组件树现在已经无需额外在main.tsx
文件中再定义了,可以直接在<body>
中完成书写。
改造通用 React 组件树入口
在原本的main.tsx
文件中,其主要功能是使用 React DOM 提供的 createRoot
函数来将 React 组件树插入到 HTML DOM 中。但是在这个示例中,由于整个组件树已经在App.tsx
中完成了向 HTML DOM 中的插入,所以main.tsx
的内容就变得十分简单,只需要引入App.tsx
中定义的文档根组件,然后进行水合处理即可。
以下是main.tsx
的全部内容。
import { hydrateRoot } from "react-dom/client";
import { App } from "./App";
hydrateRoot(document, <App />);
增加server.ts
服务器启动文件
这个服务器启动文件可以被命名为任意名称,不必局限于server.ts
,这个命名只是习惯上的。
以下是这个server.ts
文件的最小内容。
import Elysia from "elysia";
import { connect } from "elysia-connect-middleware";
import { createElement } from "react";
import { renderToReadableStream } from "react-dom/server";
import { createServer } from "vite";
import { App } from "./src/App";
const app = new Elysia();
const vite = await createServer({
server: { middlewareMode: true },
appType: "custom",
});
app.use(connect(vite.middlewares));
app.get("/", async () => {
const app = createElement(App);
const stream = await renderToReadableStream(app, {
bootstrapModules: ["/src/main.tsx"],
});
return new Response(stream, {
headers: {
"Content-Type": "text/html",
},
});
});
app.listen(8000);
console.log(
`Server running at http://${app.server?.hostname}:${app.server?.port}`
);
Elysia 服务框架的使用基本上与 Express 服务框架差不多,在这个server.ts
文件中,只有几点需要特别说明的。
- 由于 Vite 以中间件模式启动的时候,只兼容 Express 服务,所以需要使用 Elysia 的中间件插件
elysia-connect-middleware
来适配和转换 Express 的中间件供 Elysia 使用。具体 Vite 的中间件模式都做了哪些事情,可以参考 Vite 的使用文档。 - 在生产模式中,不需要以中间件模式加载 Vite。
- React Dom 提供的
renderToReadableStream
所接受的第一个参数是经过渲染的组件树,由于现在这个文件是.ts
格式,不能使用 HTML 标签,所以就只能以模块的形式导入App
组件,然后使用 React 提供的createElement
函数进行渲染。 - 用于在客户端引入的应用启动脚本
src/main.tsx
需要在renderToReadableStream
函数的第二个参数中指定,不能直接在App
组件中直接写入。React DOM 提供了多个配置项用来加载不同形式的客户端启动脚本,包括直接插入脚本内容bootstrapScriptContent
、插入脚本路径bootstrapScripts
以及插入模块bootstrapModules
的方式插入脚本。在一般情况下,插入脚本模块的方式是更加常用的,它会自动在<script>
标签中增加type="module"
属性以标识引用的脚本文件是一个模块。
有了这个文件以后,就可以在项目根目录中执行bun run server.ts
来启动支持 SSR 的 React 应用了。
如果需要实现类似于Vite的开发服务器的监控文件变更自动重载功能,可以使用bun --watch run server.ts
命令或者bun --hot run server.ts
,但是这种运行模式需要使用进程管理器(Windows)或者活动管理器(macOS)来终止。
上面这个示例中的server.ts
的内容是启动服务的最小文件,这个文件目前还只能在项目开发过程中使用。如果需要在生产环境中使用,那么就还需要引入不少的编码和配置,也就是说,如果需要server.ts
能够同时兼容开发过程使用和生产环境使用,那么server.ts
中就必须再增加能够根据当前运行环境,选择编译后文件并加载的功能。