资讯专栏INFORMATION COLUMN

使用Nuxt+Vue+Node构建的SSR博客项目

zhigoo / 2217人阅读

摘要:通过对客户端服务端基础架构的抽象组织,主要关注的是应用的渲染。文件名的名称将成为中间件名称将成为中间件。登陆成功的标志就是校验合法,然后下一步操作。校验失败就直接返回,前端根据响应跳转到登陆页面。

以前的博客使用的是Ghost,不过被攻击了,勒索我几百美元,还是算了吧,顺便说一句,数据备份很重要!前段时间学了Vue.js,以前看的Node还能记起来点,主要为了锻炼自己吧,这次的博客没有用Hexo,Ghost什么的,自己写的。由于有SEO的需求,毕竟自己写自己看也没什么意思,最终的使用的技术方案:Vue.js+Nuxt.js+ES6+Webpack+Mysql+Noyde.js+Express.js。动手之前其实还有很多东西不是很熟,不过最终也算是完成了,这个项目给了我一个启发:实践是学习的最快途径,看再多次文档真不如动手写一个项目来的实在。

Vue.js和Nuxt.js

Vue.js

当前流行的前端框架,官网地址:https://cn.vuejs.org/,文档写的很好,点个赞!

Nuxt.js

Nuxt.js的文档写的也不错,话说Vue.js系列的文档写的都很好,再次点赞,官网是这样介绍自己的:

Nuxt.js 是一个基于 Vue.js 的通用应用框架。

通过对客户端/服务端基础架构的抽象组织,Nuxt.js 主要关注的是应用的 UI渲染。

我们的目标是创建一个灵活的应用框架,你可以基于它初始化新项目的基础结构代码,或者在已有 Node.js 项目中使用 Nuxt.js。

Nuxt.js 预设了利用Vue.js开发服务端渲染的应用所需要的各种配置。

除此之外,我们还提供了一种命令叫:nuxt generate,为基于 Vue.js 的应用提供生成对应的静态站点的功能。

我们相信这个命令所提供的功能,是向开发集成各种微服务(miscroservices)的 Web 应用迈开的新一步。

作为框架,Nuxt.js 为 客户端/服务端 这种典型的应用架构模式提供了许多有用的特性,例如异步数据加载、中间件支持、布局支持等。

总结一下,Nuxt.js就是一个利用Vue, webpack 和 Node.js帮我们简单方便实现SSR的框架。

关于SSR

什么是SSR

SSR是Server Side Render的缩写,即服务器端渲染。在没有SPA之前,绝大多数的网页都是通过服务器渲染生成的:用户向服务器发送请求,服务器获取请求,然后再查询数据库,根据查询的数据动态的生成一张网页,最后将网页内容返回给浏览器端。

现在拿Vue来说,在通常情况下,Vue.js是运行在浏览器中的,在浏览器你发送一个请求,然后获得了后台返回的数据,最后通过Vue.js将数据渲染成需要的HTML片段。现在,我们把将组件渲染成HTML这个工作拿到Node上来执行,可以将Node看成一个隐形的“浏览器”,在这个“浏览器”中我们将组件都渲染好,然后将渲染好的HTML直接发送给实际的浏览器(客户端),这就是Vue SSR。

SSR的好处

通过前端框架(Vue,Angular,React)构建的SPA(Single-Page Application - 单页应用程序),因为内容是通过Ajax获取的,所以就有一个天然的缺陷,就是搜索引擎没法获取里面的内容。右键查看一个SPA网页的源代码,你会发现你里面几乎没什么内容。对于像博客,新闻这样的网站,这一点是不可接收的。

总结起来,SSR带来的好处就是能够SEO,顺便因为内容在服务器端已经渲染好了,还能够减少请求数量,对于一些比较老旧的浏览器(Vue.js不支持),也能看到基础的内容。

开发总结

开发中遇到了不少问题,在这里我列举一下,可以让大家尽量少走弯路,解决方案我尽量说的简洁点,有些东西我会多带带开一篇博文。

Restful

项目后台使用的是Restful API,后台框架用的是express,你可以用vue init nuxt-community/express-template 生成一套基于express的模板文件,后台的代码在server目录,这个目录里面就是关于Node和express的内容了,这里不展开。

asyncData多个请求

参考下面的代码:

async asyncData({ req, error }) {
    const page = 0
    let [pageRes, countRes] = await Promise.all([
        axios.get(`/api/post/page/${page}?scope=published`),
        axios.get("/api/post/count/published"),
    ])
    return {
        posts: pageRes.data.list,
        count: countRes.data.result,
    }
}

中间件

中间件允许您定义一个自定义函数运行在一个页面或一组页面渲染之前。每一个中间件应放置在 middleware/ 目录。文件名的名称将成为中间件名称(middleware/auth.js将成为 auth 中间件)。下面是一个示例:

import { isLogin } from "../util/assist"
const needAuth = require("../util/api.config").needAuth
export default function ({ isClient, isServer, route, req, res, redirect }) {
    //在服务端判读是否需要登陆(如果直接输地址,在客户端是判断不到的)
    if (isServer) {
        let cookies = req.cookies
        let path = req.originalUrl;

        if (path.indexOf("admin") > 0 && !cookies.token) {
            redirect("/login")
        }
    }
    //在客户端判读是否需要登陆
    if (isClient) {
        if (route.path.indexOf("admin") > 0 && !isLogin()) {
            redirect("login")
        }
    }
}

Node循环+异步问题

在项目里面有这样一个需求:显示文章(Post)对应的若干个标签(Tag),解决办法就是获取PostList,然后循环这个List并获取PostId,根据PostId去查对应的Tag。由于获取PostList异步操作,然后又在循环里面套了许多异步操作(获取Tag),所以用回调的方式就没法写下去了,最后使用的是async这个库,贴一段代码:

//获取Post列表
let list = (params, callback) => {
    postModel.list(params, (err, posts) => {
        if (err) {
            return callback({ code: 404, message: "no result" });
        }
        //get each posts" tags
        async.eachSeries(posts, (post, tagCallback) => {
            postTagModel.tagsByPostId(post.id, (err, result) => {
                if (err) {
                    tagCallback(err)
                }
                post.tags = result;
                tagCallback()
            });
        }, (err) => {
            if (err) {
                callback({ code: 404, message: "no result" });
                return false;
            }
            callback({ code: 404, message: "no result", list: posts });
        });
    });
}

静态资源

在Nuxt中,你可以将静态文件放到项目根目录的static文件夹中,然后直接使用根路径/就可以访问了。


登陆

第一次使用restful,登陆问题一直困扰着我,搜了不少资料,最后的解决方案是使用token。

在前端,检测到用户没有登陆就跳转到登陆页面,用户发送登陆请求后先在后台校验用户名和密码,校验成功之后返回一个token。前端接收到这个token后将它存在本地,以后每次发送请求时将这个token带上,之后的请求后台对这个token进行校验,如果合法就认为登陆成功。

下面的代码用于生成token(使用的是jwt-simple),uid是用户ID,exp的取值是七天后的时间,jwtSecret是加密和解密的密钥。

let auth = (user, callback) => {
    if (user.account.trim() == "") {
        return callback({ code: 403, message: "用户名不正确" });
    }
    if (user.password.trim() == "") {
        return callback({ code: 403, message: "密码不正确" });
    }
    userModel.auth(user, (err, user) => {
        if (err) {
            return callback({ code: 404, message: "登陆失败" });
        }
        if (user.length === 1) {
            //设置七天有效期
            let expires = moment().add(7, "days").valueOf();

            let token = jwt.encode({
                uid: user[0].id,
                exp: expires
            }, jwtSecret)

            return callback({ code: 200, message: "success", token: token });
        }
        callback({ code: 404, message: "登陆失败" });
    });
}

我使用的Ajax工具是axios,下面代码的作用是给所有请求添加一个header。

// 拦截request
$http.interceptors.request.use(
  config => {
    if (typeof document === "object") {
      let token = getCookieInClient("token")
      if (token) {
        config.headers.Authorization = token;
      }
    }
    return config;
  }, err => {
    return Promise.reject(err);
  }
);

下面是校验token的中间件,先判断接口是否需要验证身份,不需要就直接Next,如果需要就获取并校验token,校验做的比较粗糙,就是直接判断token是否在有效期内,当然可以有更安全的做法,你可以自己去搜索一下。登陆成功的标志就是token校验合法,然后下一步操作。校验失败就直接返回,前端根据响应跳转到登陆页面。

module.exports = function (req, res, next) {
    let path = req.originalUrl
    
    //接口不需要登陆:直接next
    if (needAuth.indexOf(path) < 0) {
        return next();
    }
    
    //接口需要登陆
    var token = req.headers["authorization"]
    if (!token) {
        return res.json({
            code: 401,
            message: "you need login:there is no token"
        })
    }
    
    try {
        //解密获取的token
        let decoded = jwt.decode(token, jwtSecret);

        //校验有效期
        if (decoded.exp <= Date.now()) {
            return res.json({
                code: 401,
                message: "you need login:token is expired"
            });
        }
        next();
    } catch (err) {
        return res.json({
            code: 401,
            message: "you need login:decode token fail"
        })
    }
};

nuxt.config

nuxt的配置文件,具体的配置项可以参考这个链接,下面是我的配置文件,里面的内容应该是一看就懂。

module.exports = {
  //页面的head标签
  head: {
    title: "JustYeh的前端博客",
    meta: [
      { charset: "utf-8" },
      { name: "viewport", content: "width=device-width, initial-scale=1, user-scalable=no" },
      { hid: "description", name: "description", content: "叶文祥的前端博客" },
      { name: "renderer", content: "webkit" },
      { "http-equiv": "X-UA-Compatible", content: "IE=edge" },
      { name: "author", content: "叶文祥,justyeh@163.com" },
      { name: "apple-mobile-web-app-title", content: "叶文祥的前端博客" },
      { name: "apple-mobile-web-app-capable", content: "yes" },
      { name: "apple-mobile-web-app-status-bar-style", content: "#263238" },
      { name: "screen-orientation", content: "portrait" },
      { name: "x5-orientation", content: "portrait" },
      { name: "full-screen", content: "yes" },
      { name: "x5-fullscreen", content: "true" },
      { name: "browsermode", content: "application" },
      { name: "x5-page-mode", content: "app" },
      { name: "theme-color", content: "#263238" },
    ],
    link: [
      { rel: "icon", type: "image/x-icon", href: "/favicon.ico" }
    ]
  },
  //全局引用的css文件
  css: ["~assets/css/main.css", "~assets/css/font-awesome.min.css"],
  // 页面顶部loading效果
  loading: {
    color: "#04acf7",
    height: "4px",
    failedColor: "red"
  },
  //页面的过渡效果
  transition: {
    name: "page"
  },
  //配置路由中间件
  router: {
    middleware: "adminAuth"
  }

}

部署

部署一直不怎么会弄,这次也确实在上面遇到了不少问题,这里就不说了。最终使用的是pm2,现在假设你已经安装好了node、pm2、vue等之类的包,依次运行下面的命令:

#进入文件所在目录
cd your_project
#安装项目所需依赖
npm insatll
#打包
npm run build
#运行,--name "your project name"是可选的
pm2 start npm [--name "your project name"] -- start
说在最后

上面的东西比较琐碎,如果你正好遇到相关的问题可能就有帮助了。系统性的梳理我也不知道怎么展开,只能抱歉了。

好了,全部的代码在这里:https://github.com/justyeh/justyeh.com,关于细节你都可以在这里面查看。这个是最终的成果:http://justyeh.com/,鉴于这个是私人项目,线上后台就不公布了,不过可以直接clone我的项目,数据库什么的都在,直接在本地就可以跑起来。

水平有限,如果有什么错误的地方还请包含。喜欢的话,可以给个star。如果你有什么好的建议和意见,欢迎联系我!

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/48744.html

相关文章

  • 浅谈NUXT - 基于vue.js服务端渲染框架

    摘要:是一款基于的服务端渲染框架,跟的异曲同工。该配置项用于定义应用客户端和服务端的环境变量。 Vue因其简单易懂的API、高效的数据绑定和灵活的组件系统,受到很多前端开发人员的青睐。国内很多公司都在使用vue进行项目开发,我们正在使用的简书,便是基于Vue来构建的。 我们知道,SPA前端渲染存在两大痛点:(1)SEO。搜索引擎爬虫难以抓取客户端渲染的页面meta信息和其他SEO相关信息,使...

    yearsj 评论0 收藏0
  • 浅谈NUXT - 基于vue.js服务端渲染框架

    摘要:是一款基于的服务端渲染框架,跟的异曲同工。该配置项用于定义应用客户端和服务端的环境变量。 Vue因其简单易懂的API、高效的数据绑定和灵活的组件系统,受到很多前端开发人员的青睐。国内很多公司都在使用vue进行项目开发,我们正在使用的简书,便是基于Vue来构建的。 我们知道,SPA前端渲染存在两大痛点:(1)SEO。搜索引擎爬虫难以抓取客户端渲染的页面meta信息和其他SEO相关信息,使...

    godiscoder 评论0 收藏0
  • vue服务端渲染demo将vue-cli生成项目转为ssr

    摘要:无需使用服务器实时动态编译,而是使用预渲染方式,在构建时简单地生成针对特定路由的静态文件。与可以部署在任何静态文件服务器上的完全静态单页面应用程序不同,服务器渲染应用程序,需要处于运行环境。更多的服务器端负载。 目录结构 -no-ssr-demo 未做ssr之前的项目代码用于对比 -vuecli2ssr 将vuecli生成的项目转为ssr -prerender-demo 使用prer...

    whinc 评论0 收藏0
  • vue 服务器端渲染 nuxt.js初探

    摘要:在调研插件后,发现无法满足以及等要求时,果断选用了做服务器渲染。布局目录该目录名为保留的,不可更改。服务器启动的时候,该目录下的文件会映射至应用的根路径下。它可以在服务端或路由更新之前被调用。可用于指定服务端返回的请求状态码。 开头还是来一段废话: 年关将近,给大家拜个早年,愿大家年会都能抽大奖,来年行大运。 废话不多说,直接进正文 项目环境: 前端vue项目, 需要将新增的几个路由页...

    bingchen 评论0 收藏0
  • Nuxt.js实战

    摘要:文件配置默认共用中创建标签不对标签中内容做转义处理拓展配置添加配置头部标签管理通过实现头部标签管理,在中的配置。 一、为什么选择Nuxt.js 多数是基于webpack构建的项目,编译出来的html文件,资源文件都被打包到js中,以下图404页面代码为例。从代码中可以看出,这样的页面是不利于 搜索引擎优化(SEO, Search Engine Optimization) ,并且 内容到...

    JinB 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<