SSR 框架它带着 Vite 来啦

2021-04-06 09:53:50 +08:00
 zhangyuang

SSR + Vite

在经过了大半个月的努力后终于将 Vite 接入进了我们的 SSR 框架。总共核心源码改动大概几十行代码。其实本不用这么长时间,主要是最近忙于公司业务,所以这个 PR 的代码基本都是我之前在阿里认识的校招同学贡献的,我负责出 idea 他负责代码实现。先在这里感谢一下祥杰同学的贡献。

不了解 Vite 是什么或者其工作机制的同学,可以去看看 Vite 的中文文档 或者本人之前写的 Vite 源码解析 来简单了解下。正如知乎上的问题 2021 前端会有什么新的变化? 就提到了 Vite 会取代 Webpack。今年快过半了,也看到不少团队的 PPT 分享时介绍自己用了 Vite 后构建速度从 Webpack 的 几十秒 到 Vite 的 几秒钟 的提升。可以看出 Vite 主打的优势就是快,包括闪电般的启动速度,以及 HMR 速度。

使用方式

只需以下几步便可体验成熟的 SSR + Vite 的组合
ssr-plugin-vue3 中我们将 Vite 作为一个可选配置,底层已做兼容,但默认不会安装 Vite 相关依赖。具体使用方式如下

$ npm init ssr-app my-ssr-project --template=midway-vue3-ssr # 创建 Vue3 SSR 应用,同时支持 Serverless 形式一键发布或以传统 Node.js 应用的形式部署
$ cd my-ssr-project && npm i && npm i vite @vitejs/plugin-vue --save-dev # 根据实际技术栈安装需要的插件
$ npx ssr start --vite # 建议在 package.json 中添加 "start:vite": "ssr start --vite"
$ open http://localhost:3000 # 以服务端渲染形式访问
$ open http://localhost:3000/?csr=1 # 以客户端渲染形式访问

即可使用 Vite 作为构建工具,提升启动速度和 HMR 速度。目前当前版本只在 Vue3 场景开启该功能,Vue2/React 的支持将会在下一个版本实现。详细的实现原理见下文

前言

在这里先发表一下本人对 Vite 的看法,首先从 Vite1.0 开始我便对 Vite 有着极大的兴趣。包括上述 Vite 源码解析的文档也是从 1.0 写到了 2.0 。毫无疑问从今天的角度看来,Vite 的设计和使用体验是要比 Webpack 优秀非常多的(当然其实原理完全不一样所以也不存在 Webpack 吸收 Vite 优点的可能)。但我仍然不建议盲目的抛弃 Webpack 选择 Vite。理由如下

  1. 个人认为 Webpack 最大的问题从来都不是它的性能,而是配置的复杂度过高。当然过于复杂的配置导致绝大多数人并不能正确的配置导致性能十分低下又绕回去了。但是如果正确的使用了 Webpack配置,在绝大多数场景下 Webpack > 4.0 的性能我认为是足够用的,大部分应用的启动构建速度都能够控制在 10s 以内,虽然仍然跟 Vite 的秒级启动无法相比,但也足够够用,如果你的 Webpack 项目动辄需要构建几十秒甚至一分钟以上,那么你应该思考一下是不是你的使用姿势或者是框架本身的问题。正如本篇文章介绍的 SSR 框架使用 Webpack 也用来开发过十几个页面几百个组件的应用,并没有感受到速度的明显下降。绝大多数前端框架(这里指的是在 React/Vue 上面包一层或者包很多层的那种框架比如 Next, Nuxt 等等)速度慢的原因无外乎是使用了错误的构建配置,或者是包了太多层导致自己都不知道哪一层导致性能出问题了。

  2. 旧应用迁移到 Vite 有一定成本,且无法保证稳定性。了解 Vite 的同学都知道,它是基于浏览器 ESM 特性的,虽然它提供了预优化这一步骤来将不支持 ESM 格式的第三方模块转换为 ESM 格式,但对于业务代码仍需要我们手动修改。有些几万行十几万行代码的旧应用实在是改不动,且生产环境它采用 Rollup 进行打包构建,并不能保证构建出来的行为与原 Webpack 构建出的文件行为一致。目前我的部分同事反馈会出现一些非预期的情况。这一点有待商榷需要收集整理问题。不过如果是新的 SPA 应用,倒是可以放心大胆的使用

  3. 在 SSR 场景下,本人是暂时不推荐 All in Vite 的,诚如目前 Vite 官方的 SSR Demo 也只能说自己处于试验阶段,不保证有无风险。这一点我会在接下来说明可能会出现的风险,以及我们的 SSR 框架最终采用的策略

综上所述,如果是个人应用或是新的纯客户端应用或者是体量比较小的老应用或者是当前 Webpack 应用遇到了明显的性能问题时,我觉得完全可以使用 Vite 。至于其他情况我觉得你需要想清楚究竟是否有必要使用 Vite,不要为了追求新技术而追求。毕竟公司级别的应用还是应该以稳定为主。

SSR + Vite 实现机制

注:此章节内容,如果是之前未了解过服务端渲染原理的同学可能阅读起来会有一定障碍。

在 Vite 官方给出了一个简单的 SSR Demo 后,我就去看了一下。代码非常简单易懂,阅读起来难度不大。首先是服务端打包入口客户端打包入口文件这一块,其实与原来的 Webpack 场景几乎没有任何区别可以直接移植。值得关注的点如下

Vite MiddleWare

如下代码所示

if (!isProd) {
    vite = await require('vite').createServer({
      root,
      logLevel: isTest ? 'error' : 'info',
      server: {
        middlewareMode: true
      }
    })
    // use vite's connect instance as middleware
    app.use(vite.middlewares)
  } else {
    app.use(require('compression')())
    app.use(
      require('serve-static')(resolve('dist/client'), {
        index: false
      })
    )
  }

Vite 在本地开发时,提供了中间件的形式可以在任何 Node.js 框架接入。默认的示例使用的是 express。此中间件的作用是接管客户端文件的请求。相比于 Webpack 构建出来的 bundle 执行流程的区别如下

相比于使用 Webpack 我们少了构建 bundle 的操作,来提升应用的启动速度和 HMR 速度 在生产环境时,Vite 则使用 Rollup 来将代码打包成一个 bundle 放在静态资源文件夹中加载,这种形式执行起来就跟 Webpack 没有任何区别了。只有在本地开发时才会感受到明显区别。

Vite LoadModule

此外,Vite 还提供了 LoadModule API 来解析 entry-server, 使得我们可以直接在 Node.js 环境执行 ESM 格式的代码,且支持 HMR

render = (await vite.ssrLoadModule('/src/entry-server.js')).render

分析完之后其实一个 Vite SSR 应用的执行流程就很容易看懂了。但是我们的框架并没有采用跟官方一样的方案思路。理由如下

SSR 框架升级策略

下面介绍了一下本框架是如何最小成本化的接入 Vite 的,如果有不准确的地方可以提 issue 或者私聊会及时的进行更正。

首先在 Vite 出现大部分类似的框架都是使用 Webpack 来作为构建工具,如果全量切换成 Vite 或者将 WebpackVite 完全隔离以插件的形式来让用户决定到底是用 Webpack 还是 Vite 进行构建,那么改动量我认为前者会非常大,后者的工作量也不小。但是其实在我们的框架完全分离其实也不难,因为我们的核心源码也就几千行,不像 Next, Nuxt 这种代码巨无霸级别的项目就算支持了 Vite 我认为 Bug 也会非常多。但是即使是这样,在这个版本中,我们仍没有打算完全的分离 WebpackVite,理由如下

由于上述原因,在这个版本中我定下的接入 Vite 方案如下

思考了一下此方案的优劣势如下

优势

不足

其实这种开箱即用的框架其实我都不建议开放类似于 chainWebpack 之类的配置来让用户去自己 diy 。可以看看 github 绝大部分的 issue 都是由于用户自己 diy 引起的。要么默认的配置就做到足够好,要么只提供关键配置让用户使用例如 cssloaderOptions,而不是整个 chainWebpack 这样的 API

性能对比

由于我们仍然使用了 Webpack 去构建服务端 bundle 所以注定我们的性能速度跟 All in Vite 的情况肯定还是存在一定差异的。但其实就如同上面提到的,Webpack 的最大问题从来不是性能并且本框架之前的 Webpack 配置性能已经足够使用了,特别是在打包服务端 bundle 这一块,由于我们开启了 externals 配置需要打包的文件体积非常小,所以我们的打包速度大概在 2s 内即可完成。

改造后使用 Vite 启动的构建步骤如下

  1. 启动命令 import 一些 必要的模块,这里我们已经对 Webpack 的依赖做了细致的分离,例如在 Vite 场景下很多 Webpack 打包客户端 bundle 所需要的依赖我们是没有必要加载的,这样会拖慢我们的启动速度。

  2. Webpack 构建服务端 bundle (耗时在 2s 左右)

  3. Midway/Nest.js 服务启动,加载 Vite 中间件 (耗时在 3s 左右)

以图表的形式展示的话如下

跟之前版本的做法比起来。在启动速度上,我们少了构建客户端 bundle 这一步,时间大概能减少 2s 左右当应用变得庞大时这块的时间差距应该会更加明显, HMR 速度则提升的更加明显,主要是之前的 Webpack 版本构建速度已经足够快了。如果你之前使用的是 Nuxt, Next 这种项目稍微一大起来速度就慢的跟蜗牛一样的巨无霸框架那你应该能感受到明显的速度提升。

All in Vite 的方式比起来,我们多了一个 Webpack 构建服务端 bundle 的耗时。至于 Midway/Nest.js 服务启动,加载 Vite 中间件这一步无论你使用什么方案这都是必经之路这块的速度可以让 Midway 的同学优化一下。

上点实际图片可以更直观的感受。

使用 Webpack 启动应用

以默认创建并发布到 Serverless 的 example 作为示例,该示例包含首页,详情页。使用了 vue-router, vuex, swiper 等基础组件,虽然复杂度比不上真实应用,但涉及到的功能和真实线上应用区别不大。使用 Webpack 构建

可以看出完整启动时间大概在 6s 左右

使用 Vite 启动应用

启动时间缩短到了 3 秒左右,并且此时我们的前端页面行为都被 Vite 接管。比如当我们修改前端组件时可以看到终端 Vite HMR 的提示

浏览器中的 Vite Info

总结

写了这么多内容,其实也并不代表这里介绍的方案是最优的,但是我认为肯定是修改成本最小的方案。主要还是想跟大家分享一下我在升级过程中的一些思考。代码实现从来都不是最难的,难的是思路以及整体层面的一些思考设计。就像我在做 SSR 场景下代码分割的一些想法,也跟官方 loadable 的推荐做法完全不一致。在这里感谢一下 尤大的贡献让大家的工作效率提升这么多可以提早下班,有些同学可能看了点 Vite 的源码或者 Vue 源码后觉得不难感觉自己也能实现。其实这是错误的思想。在这里引用一段回答

以王垠的水平,他可以开发出像 nginx 和 redis 这样的软件吗? 答案:nginx 和 redis 的伟大之处, 不在于它们技术有多难,基本上任何一个合格的程序员,在学习它们背后的设计思路和底层技术后, 通过记忆复刻一个类似的东西出来并不算难。nginx 和 redis 之所以伟大, 是它们都是它们的作者出于对现实的不满意,通过自己的能力改造世界的成果。这背后最不可思议的是,一个人能认识到现实世界中的不完美, 同时他还能想到一种方式做出改进,同时他还有能力实现自己的想法,这三件事巧合地发生在一个人身上,这才是最难能可贵的。

最后的最后,也并不排除之后我们的SSR 框架会将 Webpack/ Vite 完全分离,提供 All in Vite 的方案给大家使用我们会随时跟进 Vite 的最新进展,因为这套方案已经在微信内部正式的投入使用了,我们会持续更新维护下去。其实这个框架从 1.0 egg-react-ssr 的版本开始到今天也算开发了 2 年 多了。作者都已经从 阿里 跳到 微信了,绝不是什么 KPI 项目纯属个人对 coding 的爱好支撑维护欢迎大家使用。如果你觉得这篇文章对你有帮助,麻烦点个 Star✨

1725 次点击
所在节点    程序员
9 条回复
Cooky
2021-04-06 10:19:44 +08:00
从屎山跳到另一坨屎山
zhwithsweet
2021-04-06 10:31:51 +08:00
@Cooky 用 vite or snowpack 了?就屎山?
jmyz0455
2021-04-06 10:39:02 +08:00
感谢有人在现有的限制下(被人诟病的 js 和 node 工具们、各有实现的浏览器、停不下来的框架战争),去想办法优化和提高前端的开发效率。
gdrk
2021-04-06 10:43:35 +08:00
请问下现有 vue2 + element 的项目迁移到你们的 ssr 框架是不是会比迁移到 nuxt 快,学习成本相对会低一些么
Cooky
2021-04-06 11:52:28 +08:00
@zhwithsweet 阿里屎山 --> 微信屎山
int64ago
2021-04-07 10:05:21 +08:00
发一遍就差不多了,天天这里发没用的。

不过看你也毕业没多久,能做到并落地这些也挺棒了。

给点切实的建议吧:

1. 完善下项目文档,补充英文文档;
2. 约束下使用场景,Next 和 Nuxt 并没有那么不堪,你要刚就先需要圈定场景;
3. 不要承诺会持续更新,特别是这种几乎只有一个人在更新的项目;
4. 推广文和技术文要有一定边界,不要混在一起,要推广就说特性、说解决的问题、说对比优势等,要技术布道就单纯说技术,最后顺带个推广即可。


我看过太多一开始因为一时激情投入,随着激情退却,项目沦为屎坑的例子了,希望你可以坚持久点吧。

我个人认为一个靠谱的开源项目一定要有商业介入的,无论是投人还是投钱。
zhangyuang
2021-04-26 14:19:29 +08:00
@Cooky 但凡你有点水平也说不出这种话
zhangyuang
2021-04-26 14:21:15 +08:00
@int64ago 1 、项目文档完爆一堆开源项目
2 、正式写还是要谦虚一点,当然除了我之外认为 next nuxt 烂的人一大把
3 、从 1.0 版本开始已经更新维护 3 年了,交流群近 1000 人,大规模应用几十个
4 、这只是框架的一个技术点而已
5 、next 那种十几万行代码的才能叫屎坑,4000 行的怎么也沦不了
zhangyuang
2021-04-26 14:23:03 +08:00
@Cooky 水平低就多向我学习点

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/768252

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX