欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Nuxt配合Node在实际生产中的应用详解

程序员文章站 2022-05-15 12:22:13
上个星期,甲方说要在应用上做一个促活活动(其实就是让用户领ofo的骑车券),考虑到之前在我另一个应用上已经做过客户端token验证,想想,撸起袖子直接在之前的node上开搞...

上个星期,甲方说要在应用上做一个促活活动(其实就是让用户领ofo的骑车券),考虑到之前在我另一个应用上已经做过客户端token验证,想想,撸起袖子直接在之前的node上开搞吧。

nuxt

这个东西类似与react中的next,做vue服务端渲染在nuxt确实挺方便的,整个目录结构与vue-cli也差不多,但我觉得更清晰明了,而且有很丰富的配置,包括我最喜欢的顶部loadding条啊、layer主题模板啊、服务端渲染有益于seo啊、自动生成router啊等等等等。这里我就不详细展开了,选择他主要是用习惯了,而且性能也很不错。

express

它就是一个koa、egg等的一个服务端的工具,封装很多便捷的方法。这里也不详细展开。

第一步 完成静态页面

在这里就不详细说明html怎么构建、vue使用方法等等等不相关的问题,首先页面效果如下图:

Nuxt配合Node在实际生产中的应用详解 

点击确认领取后,向后端发送请求,获取券码:

Nuxt配合Node在实际生产中的应用详解 

大致业务流程就是这样子了。

第二步 基于webview完成token验证

甲方公司的token是怎么给到用户的呢?

  1.  用户登录客服端
  2. 户在webview访问相应的地址
  3. 客户端把token等信息绑定在地址上让服务端接受,并向服务端发送get请求
  4. 服务端接受到token后,根据token信息验证用户身份并返回页面给用户

所以也就是相当于,此时我手上的node需要完成一个token验证的需求,开搞:

这里必须说一下session这个东西,一开始当我完成了token验证的时候,可是发现两个用户同时访问的时候,后访问者会重新执行一次对应请求里的方法,相当于很多数据都初始化甚至重新赋值。完全不知道怎么把用户分开,问了一下java的同学,他们说框架自带会分开,这就让我很忧郁了。其实,他们也知道session这个东西,只是一直这样用后习惯了也就突然说不清楚甚至想到它的存在了。

我一开始想到的是用cookie验证对应浏览器,但总觉得这种方法有点古老,不够高大上(个人主观看法,勿喷),储存量也很有限。所以又探索了一天,得知有个session这个东西可以验证对应浏览器(好吧,其实也就是把cookie当成数据索引)。知道了有这么一回事之后也就茅塞顿开了。

// 引入一些工具
const http = require('http')
const express = require("express");
const session = require('express-session');

const nuxtapp = express(); // 创建一个express应用

// 设置session配置
nuxtapp.use(session({
 secret: 'key' // 建议使用 128 个字符的随机字符串 
}));

// 处理options请求(axios发送post请求时会先发送一个option请求验证服务器的连通情况)
nuxtapp.use(function (req, res, next) {
 if ('options' === req.method) {
  res.sendstatus(200);
 } else {
  next();
 }
});

// 写一个验证token的接口
nuxtapp.get('/getphone', (req, res) => {
  if (!req.query.token) {
   // if else ...
   res.sendstatus(200)
   return;
  }
  // 这里创建请求只是举一个例子,有的token可以本地解析,有的需要服务端向另一个服务端请求解析。(参考微信)
  const request = http.request({
   host: "api.example.com",
   headers: {
    'content-type': ' application/json',
    'accept-encoding': 'utf-8', //这里设置返回的编码方式 设置其他的会是乱码
    'accept-language': 'zh-cn,zh;q=0.8',
    'connection': 'keep-alive',
   },
   path: '/gettokenorsth',
   port: 'port',
   method: 'post'
  }, response => {
   let data = ""; // 创建变量记录数据
   response.on("data", function (chunk) {
    data += chunk
   })
   response.on("end", function () { // 设置seesion
    try {
     if (!req.session.id) {
      // set session like : req.session.id= json.parse(data)
     }
     res.sendstatus(200)
    } catch (err) {
     console.log(err);
     res.sendstatus(500)
    }
   })
  }).on('error', (e) => {
   console.log(`错误信息: ${e.message}`);
   res.sendstatus(500)
  });
  // post data
  request.write(json.stringify({
    example:example,
    token: req.query.token
  })); // 用户传过来的数据 post
  request.end();
})

nuxtapp.listen(port || 80, '0.0.0.0')
console.log("已开启服务器,请访问 —— localhost:" + port || 80)

第三步 服务端代理(这里我也有个点不是很确定,所谓淘宝node+java是不是就是这种模式)

毕竟我还是个前端,功能方面的实现还是交给资深大java吧(若是给我时间,我还真的想写一下sql复习一下)

用户访问的时候还是向node请求,这里需要提一点的就是,假如后端提供了很多接口,你不一定每个都要写一个app.get(‘/api')或者什么,express的文档上有提到类似这种写法:

nuxtapp.all(["/api1/*","/api2/*"], requestfunction)

其他就跟上面token验证差不多了

第四步 用express执行nuxt

在nuxt官网上其实也有部分代码教你如何用自己写的node去运行,但一直不是很完善,有点差强人意。我在这里就补充一下我的理解, 哪里写得不对,请尽管喷,谢谢喷我的每一个人

// nuxt
const {
 nuxt,
 builder
} = require('nuxt') // 引入核心构建属性

// 判断开发环境
const isprod = (process.env.node_env === 'production')
const port = process.env.port || 80

// 引入nuxt配置
const config = require('./nuxt.config.js')
config.dev = !isprod;
const nuxt = new nuxt(config);

// 判断生产模式 dev(开发者模式)表示重新构建 ;pro(生产模式)表示直接从yanr build的文件直接执行
if (config.dev) {
 new builder(nuxt).build()
  .then(listen)
  .catch((error) => {
   console.error(error)
   process.exit(1)
  })
} else {
 listen()
}

以上的listen就是我在第二、三步写的相应接口的express服务器啦。

那么nuxt其实还并没有在代码中执行,new nuxt一个nuxt实例后返回一个nuxt方法,它其实是根据request,response去执行相应的渲染,其中我们要考虑到express()的各种接口的执行顺序,假如一开始就匹配到了第二步的代码的get请求,那么写在第二步代码后的all方法中的各种接口是接收不到的。

我考虑到的是接口不是无缘无故执行的,但用户请求是必然发生的。那么也就是当用户除了故意发送请求,全部由nuxt接手。所以nuxt会是放在最后让其渲染页面并交给用户,为了方便验证token,我把验证token的请求放在首页让浏览器捕获token以及相应信息后交给node执行手动登陆。(其实也可以通过在nuxt渲染前判断是否存在token来决定是否进行token验证,我就不在这里再进行大量代码的展示了)

结合以上思路,也就是在最后捕获所有get请求并交手给nuxt即可:

// 最后捕获nuxt渲染
nuxtapp.get('/*', (req, res) => {
 nuxt.render(req, res)
})

demo:

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。