经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » JSJS库框架 » JavaScript » 查看文章
为我们的SSR程序添加热更新功能
来源:cnblogs  作者:不懂代码的攻城师  时间:2018/10/16 9:29:33  对本文有异议

前沿

通过上一篇文章 通过vue-cli3构建一个SSR应用程序 我们知道了什么是SSR,以及如何通过vue-cli3构建一个SSR应用程序。但是最后遗留了一些问题没有处理,就是没有添加开发时的热更新功能,难道要每次更新代码都要重新编译打包吗?显然不是很合理。那接下来我们将为该SSR程序添加热更新的功能。

1、解决思路

我们知道SSR程序每次打包编译完成后,都会生成这两个文件 vue-ssr-client-manifest.jsonvue-ssr-server-bundle.json

  • vue-ssr-client-manifest.json

主要记录了静态资源文件的配置信息

  • vue-ssr-server-bundle.json

主要记录了js文件的内容

那现在就是要解决如何在保存代码后,获取到最新的vue-ssr-client-manifest.jsonvue-ssr-server-bundle.json这两个文件。

通过该图,我们知道,既然要热更新,那 webpack dev server 肯定跑不了。

所以解决的步骤如下:

  1. 起一个webpack dev server 服务,暴露8080端口
  2. 起一个webpack compiler 编译webpack配置文件,监听文件修改,实时编译获取最新的 vue-ssr-server-bundle.json
  3. 通过webpack dev server 获取最新的 vue-ssr-client-manifest.json
  4. 结合 vue-ssr-server-bundle.json 和 vue-ssr-client-manifest.json 渲染html页面返回给浏览器

2、编码实现

有了思路后,剩下的就是要思考如何通过代码实现了。

2.1、 起一个webpack dev server 服务

通过 npm run serve 我们能很快的起一个webpack dev server 服务

  1. npm run serve

2.2、获取webpack配置文件,并编译

通过阅读官方文档我们知道webpack的配置文件在 /node_modules/@vue/cli-service/webpack.config.js 中

  1. // 1、webpack配置文件
  2. const webpackConfig = require('@vue/cli-service/webpack.config')

2.3、编译webpack配置文件,并监听文件修改

  1. // 2、编译webpack配置文件
  2. const serverCompiler = webpack(webpackConfig)
  3. const mfs = new MemoryFS()
  4. // 指定输出到的内存流中
  5. serverCompiler.outputFileSystem = mfs
  6. // 3、监听文件修改,实时编译获取最新的 vue-ssr-server-bundle.json
  7. let bundle
  8. serverCompiler.watch({}, (err, stats) =>{
  9. if (err) {
  10. throw err
  11. }
  12. stats = stats.toJson()
  13. stats.errors.forEach(error => console.error(error) )
  14. stats.warnings.forEach( warn => console.warn(warn) )
  15. const bundlePath = path.join(
  16. webpackConfig.output.path,
  17. 'vue-ssr-server-bundle.json'
  18. )
  19. bundle = JSON.parse(mfs.readFileSync(bundlePath,'utf-8'))
  20. console.log('new bundle generated')
  21. })

2.4、获取最新的 vue-ssr-client-manifest.json

  1. // 4、获取最新的 vue-ssr-client-manifest.json
  2. // 这边的 8080 是 dev server 的端口号
  3. const clientManifestResp = await axios.get('http://localhost:8080/vue-ssr-client-manifest.json')
  4. const clientManifest = clientManifestResp.data

2.5、结合各个步骤的核心后的最后代码

安装所需要的库

  1. npm install webpack memory-fs concurrently -D
  2. npm install koa-router axios -S

在项目根目录下 新建一个 server/dev.ssr.js,代码如下

  1. // server/dev.ssr.js
  2. const webpack = require('webpack')
  3. const axios = require('axios')
  4. const MemoryFS = require('memory-fs')
  5. const fs = require('fs')
  6. const path = require('path')
  7. const Router = require('koa-router')
  8. // 1、webpack配置文件
  9. const webpackConfig = require('@vue/cli-service/webpack.config')
  10. const { createBundleRenderer } = require("vue-server-renderer");
  11. // 2、编译webpack配置文件
  12. const serverCompiler = webpack(webpackConfig)
  13. const mfs = new MemoryFS()
  14. // 指定输出文件到的内存流中
  15. serverCompiler.outputFileSystem = mfs
  16. // 3、监听文件修改,实时编译获取最新的 vue-ssr-server-bundle.json
  17. let bundle
  18. serverCompiler.watch({}, (err, stats) =>{
  19. if (err) {
  20. throw err
  21. }
  22. stats = stats.toJson()
  23. stats.errors.forEach(error => console.error(error) )
  24. stats.warnings.forEach( warn => console.warn(warn) )
  25. const bundlePath = path.join(
  26. webpackConfig.output.path,
  27. 'vue-ssr-server-bundle.json'
  28. )
  29. bundle = JSON.parse(mfs.readFileSync(bundlePath,'utf-8'))
  30. console.log('new bundle generated')
  31. })
  32. // 处理请求
  33. const handleRequest = async ctx => {
  34. console.log('path', ctx.path)
  35. if (!bundle) {
  36. ctx.body = '等待webpack打包完成后在访问在访问'
  37. return
  38. }
  39. // 4、获取最新的 vue-ssr-client-manifest.json
  40. const clientManifestResp = await axios.get('http://localhost:8080/vue-ssr-client-manifest.json')
  41. const clientManifest = clientManifestResp.data
  42. const renderer = createBundleRenderer(bundle, {
  43. runInNewContext: false,
  44. template: fs.readFileSync(path.resolve(__dirname, "../src/index.temp.html"), "utf-8"),
  45. clientManifest: clientManifest
  46. });
  47. const html = await renderToString(ctx,renderer)
  48. ctx.body = html;
  49. }
  50. function renderToString(context,renderer) {
  51. return new Promise((resolve, reject) => {
  52. renderer.renderToString(context, (err, html) => {
  53. err ? reject(err) : resolve(html);
  54. });
  55. });
  56. }
  57. const router = new Router()
  58. router.get("*", handleRequest);
  59. module.exports = router

新建一个 server/ssr.js,代码如下

  1. // server/ssr.js
  2. const Koa = require('koa')
  3. const koaStatic = require("koa-static");
  4. const path = require('path')
  5. const resolve = file => path.resolve(__dirname, file);
  6. const app = new Koa()
  7. const isDev = process.env.NODE_ENV !== 'production'
  8. const router = isDev ? require('./dev.ssr') : require('./server')
  9. app.use(router.routes()).use(router.allowedMethods())
  10. // 开放目录
  11. app.use(koaStatic(resolve("../dist")));
  12. const port = process.env.PORT || 3000;
  13. app.listen(port, () => {
  14. console.log(`server started at localhost:${port}`);
  15. });
  16. module.exports = app

修改package.json 添加几个执行脚本

  1. // package.json scripts字段
  2. "dev:serve": "cross-env WEBPACK_TARGET=node node ./server/ssr.js",
  3. "dev": "concurrently \"npm run serve\" \"npm run dev:serve\" "

执行 npm run dev 命令

  1. npm run dev

访问 localhost:3000 你会发现,还是有问题。

静态资源的文件引用的是 node.js server 的服务的,即引用到了3000端口上去了,但是3000端口的服务并没有这些静态资源文件,
这些静态资源文件在webpack dev server中。

那如何解决呢?

  1. node server将这些静态资源的请求代理到 webpack dev server中
  2. 改变webpack的baseUrl,直接引用到webpack dev server中

很显然,第二种方式实现起来比较简单,那我们就修改webpack的baseUrl配置

修改 vue.config.js

  1. // vue.config.js
  2. // 添加一个字段,如果是开发环境,就指定到webpack dev server中
  3. baseUrl: isDev ? 'http://127.0.0.1:8080' : '',

重新 npm run dev ,然后访问 localhost:3000

已经能正常访问了,那我们试试 热更新的更新能不能实现,修改一段代码,

会发现 node server 会重新编译webpack配置文件,然后在看看浏览器有没有更新

你会发现浏览器还是没能热更新内容,打开 F12,你会发现这个错误,

这是我们常见的不允许跨域的错误提示。

解决方式:

  1. 配置 webpack dev server 允许跨域
  1. // vue.config.js
  2. // 添加一个 devServer的字段
  3. devServer: {
  4. headers: {'Access-Control-Allow-Origin': '*'}
  5. },

重新 npm run dev ,然后访问 localhost:3000

是已经能实现热更新的了。

3、优化

1、favicon的问题
打开f12还是能看到有问题,

具体实现可以参考我的github代码

2、修改 server 端代码自动重启代码

可以使用nodemon,或者pm2实现

4、总结

通过上一篇 通过vue-cli3构建一个SSR应用程序 和这篇文章,我们一步一步搭建起了基于vue-cli3的一个ssr应用程序,并添加了热更新的功能,在这期间也踩了很多坑。但是最终实现了之后,你会觉得这些付出都是值得的,因为这些都是自己的成长奠定基础。

如果有更好的实现方法,欢迎交流交流!

如果有不对的地方,欢迎指出!

5、源码

项目源码:vue-cli-ssr-example 欢迎 star

赞赏

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号