SPA单页应用前后分离微信授权
项目基于微信公众号开发,业务完全依赖微信授权,也就是用户进入页面已经完成授权获取到用户的OpenId。
需要有一个授权中间页:author.vue
基本实现思路:
- 无论使用哪个url进入页面都会先触发router.beforeEach钩子。
 - 在router.beforeEach钩子函数中判断用户是否授权。
 - 若未授权则保存用户进入的url并请求后台接口获取微信授权(window.location.href=‘后台接口’)。
 - 后台调用微信接口授权获取用户信息及openId,将openId使用JWT生成一个唯一的token令牌,并将token已参数的形式拼接到url后面,然后重定向到前端author.vue页面。
 - author页面获取url中的token参数,将token参数保存到本地缓存。
 - 获取签名用户保存的url并跳转。
 
前端代码实现:
路由index.js
// 全局守卫,微信授权
router.beforeEach((to, from, next) => {
// 路由发生变化修改页面title
if (to.meta.title) {
document.title = to.meta.title
}
if (process.env.NODE_ENV !== 'development') {
const token = window.localStorage.getItem('token')
if (token) {
if (to.path === '/author') {
next({
path: '/'
})
} else {
next()
}
} else {
if (to.path !== '/author') {
// 保存用户进入的url
window.localStorage.setItem('authUrl', to.fullPath)
// 跳转到微信授权页面
window.location.href = process.env.BASE_URL + '/wx/OAuth2/index'
} else {
next()
}
}
} else {
window.localStorage.setItem('token', 'eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJhdWQiOiJvUUFFYndSSU5VVlhPLVZoOWhEcDUzX3RNeEgwIn0.eShRG4fVFFv4w2gHnkyh7QDdVpG1meOHSZXOrbq-psE')
}
next()
})
Author.vue
<template>
<div>授权中</div>
</template> <script>
export default {
name: 'Author',
data () {
return {
user: null
}
},
created () {
// url中获取参数token
const wxToken = this.$route.query.token
// url中获取参数code
const code = this.$route.query.code
// 后端重定向获取参数,判断是否处理成功 200:成功
if (wxToken && Number(code) === 200) {
// 将token放入本地缓存
window.localStorage.setItem('token', wxToken)
// 从本地缓存中获取用户第一次请求页面URL
const historyUrl = window.localStorage.getItem('authUrl')
// 跳转页面
this.$router.push(historyUrl)
} else {
// 没有拿到后台访问微信返回的token
// 清空本地缓存
window.localStorage.removeItem('token')
window.localStorage.removeItem('authUrl')
}
}
}
</script> <style scoped> </style>
后端代码实现:
/**
* 微信授权 --- OATH2 -- 第一种方式(推荐)
* 第一步:前端请求-/wx/oAth2/index
* 第二步:重定向-微信服务器
*/
@PassToken
@GetMapping(value = "/wx/OAuth2/index")
public void OAth2(HttpServletResponse response) throws IOException{
response.sendRedirect(wxMpService.oauth2buildAuthorizationUrl(baseUrl + "/wx/OAuth2/redirect",
WxConsts.OAuth2Scope.SNSAPI_USERINFO, null));
} /**
* 微信授权 -- 微信回调
* 第一步:获取code
* 第二步:通过code获取用户信息
* 第三步:Jwt生成Token令牌
* 第四步:重定向 --> 前端页面
*/
@PassToken
@GetMapping(value = "/wx/OAuth2/redirect")
public void OAth2Return(HttpServletRequest request, HttpServletResponse response) throws IOException,WxErrorException{
String code = request.getParameter("code");
// 获取用户信息
WxMpUser wxMpUser = wxMpService.oauth2getUserInfo(wxMpService.oauth2getAccessToken(code), null);
log.info("[微信授权]--------拉取用户信息详细如下:{}",wxMpUser);
//将微信用户信息入库
wxUserInfoService.insertWxUser(wxMpUser);
//生成token令牌
String token = JWT.create().withAudience(wxMpUser.getOpenId()).sign(Algorithm.HMAC256(jwtSecret));
//重定向地址
String redirectUrl = frontUrl + "/#/author" + "?token=" + token + "&code=200";
response.sendRedirect(redirectUrl);
}
后台验证用户信息
前端获取到token令牌之后,前端每次请求,后端如何获取OpenId以及业务处理?
基本实现思路:
- 前端使用axios请求拦截器,判断本地缓存是否存在token,如果存在的话,则为每个Http请求赋值token。
 - 后端使用拦截器拦截有@PassToken注解以外的方法,获取token值。如果token为null,直接返回错误码以及错误信息。
 - 验证token值是否有效,如有效,则解析openId,并将openId放入request中放行。如无效,直接返回错误码以及错误信息。
 - 拦截器放行,后端可直接通过request.getAttribute("openId")获取。
 
前端代码实现:
request.js
// 请求拦截器
axios.interceptors.request.use(function (config) {
config.headers['Content-Type'] = 'application/json;charset=UTF-8'
// 判断本地缓存是否存在token,如果存在的话,则每个http header都加上token
if (window.localStorage.getItem('token')) {
config.headers.authorization = window.localStorage.getItem('token')
}
return config
}, function (error) {
return Promise.reject(error)
})
后端代码实现:
JwtInterceptor.java
public boolean preHandle(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object object) throws Exception {
        // 从 http 请求头中取出 token
        String token = httpServletRequest.getHeader("Authorization");
        // 如果不是映射到方法直接通过
        if(!(object instanceof HandlerMethod)){
            return true;
        }
        HandlerMethod handlerMethod=(HandlerMethod)object;
        Method method=handlerMethod.getMethod();
        // OPTIONS请求类型直接返回不处理
        if ("OPTIONS".equals(httpServletRequest.getMethod())){
            return false;
        }
        //检查是否有passToken注释,有则跳过认证
        if (method.isAnnotationPresent(PassToken.class)) {
            PassToken passToken = method.getAnnotation(PassToken.class);
            if (passToken.required()) {
                return true;
            }
        }
        //校验token,并且将openId放入request中
        if (StrUtil.isNotEmpty(token)){
            // 验证 token
            JWTVerifier jwtVerifier = JWT.require(Algorithm.HMAC256(jwtSecret)).build();
            try {
                jwtVerifier.verify(token);
            } catch (JWTVerificationException e) {
                logger.info("token校验未通过");
                httpServletResponse.getWriter().println(JSONUtil.toJsonStr(Result.need2BLogged()));
                return false;
            }
            // 获取 token 中的 openId
            String openId;
            try {
                openId = JWT.decode(token).getAudience().get(0);
                httpServletRequest.setAttribute("openId",openId);
            } catch (JWTDecodeException j) {
                throw new RuntimeException("401");
            }
        }
        //检查有没有需要用户权限的注解
        if (method.isAnnotationPresent(UserLoginToken.class)) {
            UserLoginToken userLoginToken = method.getAnnotation(UserLoginToken.class);
            if (userLoginToken.required()) {
                // 执行认证
                if (token == null) {
                    throw new RuntimeException("无token,请重新登录");
                }
                // 获取 token 中的 openId
                String openId;
                try {
                    openId = JWT.decode(token).getAudience().get(0);
                } catch (JWTDecodeException j) {
                    throw new RuntimeException("401");
                }
                // 通过 openId 查询用户是否绑定手机号
                if (objectRedisTemplate.hasKey(userIdKey + openId)) {
                    logger.info("通过FRDIES用户拦截器");
                    return true;
                } else {
                    logger.info("REDIS:{Redis has no user information}");
                    //根据 openId 查询该用户的信息
                    BaseUserInfo userInfo = baseController.getUserInfo(httpServletRequest, httpServletResponse);
                    if (userInfo != null && StrUtil.isNotEmpty(userInfo.getPhone())){
                        logger.info("通过用户拦截器");
                        return true;
                    }else{
                        // 未绑定手机用户返回
                        httpServletResponse.getWriter().println(JSONUtil.toJsonStr(Result.need2BLogged()));
                        return false;
                    }
                }
            }
        }
        return true;
    }
@PassToken
package com.yhzy.zytx.jwt.annotation; import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target; /**
* @ClassName PassToken
* @Description 自定义注解(跳过验证Token)
* @Author 天生傲骨、怎能屈服
* @Date 2019/5/22 13:38
* @Version 1.0
*/
@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
public @interface PassToken {
boolean required() default true;
}
到这里整个前后分离微信授权的流程就完了,希望可以帮助到大家!!!
SPA单页应用前后分离微信授权的更多相关文章
- 前端 SPA 单页应用数据统计解决方案 (ReactJS / VueJS)
		
前端 SPA 单页应用数据统计解决方案 (ReactJS / VueJS) 一.百度统计的代码: UV PV 统计方式可能存在问题 在 SPA 的前端项目中 数据统计,往往就是一个比较麻烦的事情,Re ...
 - [vue]spa单页开发及vue-router基础
		
- 了解spa页面跳转方式:(2种) spa: 单页跳转方式 开发(hash模式): https://www.baidu.com/#2313213 生产(h5利于seo): history.pushS ...
 - Javascript 与 SPA单页Web富应用
		
书单推荐 # <单页Web应用:JavaScript从前端到后端> http://download.csdn.net/detail/epubitbook/8720475 # <MVC ...
 - 【读书笔记】WebApi 和 SPA(单页应用)--knockout的使用
		
Web API从MVC4开始出现,可以服务于Asp.Net下的任何web应用,本文将介绍Web api在单页应用中的使用.什么是单页应用?Single-Page Application最常用的定义:一 ...
 - 大熊君学习html5系列之------History API(SPA单页应用的必备)
		
一,开篇分析 Hi,大家好!大熊君又和大家见面了,(*^__^*) 嘻嘻……,这系列文章主要是学习Html5相关的知识点,以学习API知识点为入口,由浅入深的引入实例, 让大家一步一步的体会" ...
 - 基于VUE的SPA单页应用开发-加载性能篇
		
1.基于异步数据的vue页面刷新 先看看基于异步数据的vue页面刷新后,都发生了啥- 如图所示: 图1 基于异步数据的vue页面刷新 网络请求图 步骤如下: step1:请求页面: step2:请求页 ...
 - 前端学习之路之SPA(单页应用)设计原理
		
SPA设计 1.设计意义 前后端分离 减轻服务器压力 增强用户体验 Prerender预渲染优化SEO 前后端分离:前端做业务逻辑,后端处理数据和接口,耦合度减少,开发效率提高. 减轻服务器压力:一个 ...
 - 大熊君学习html5系列之------History API(SPA单页应用的必备------重构完结版)
		
一,开篇分析 Hi,大家好!大熊君又和大家见面了,(*^__^*) 嘻嘻……,这系列文章主要是学习Html5相关的知识点,以学习API知识点为入口,由浅入深的引入实例, 让大家一步一步的体会" ...
 - 3.【nuxt起步】-下面以一个SPA单页程序为例子
		
spa:single page applcation 1.components目录新建header.vue,footer.vue Header.vue Footer.vue Pages/index.v ...
 
随机推荐
- JS中继承方式总结
			
说在前面:为了使代码更为简洁方便理解, 本文中的代码均将"非核心实现"部分的代码移出. 一.原型链方式关于原型链,可点击<深入浅出,JS原型链的工作原理>,本文不再重复 ...
 - HTML5/CSS3淡入淡出滑块焦点图
			
在线演示 本地下载
 - 《CSS权威指南(第三版)》---第四章 值和单位
			
本章主要讲解的是一些属性声明用的值: CSS中的值主要有数字,百分数,颜色, 1.颜色: rgb(100%,100%,100%) OR rgb(255,255,255) OR #FF0000 WE ...
 - 同级别中枢重叠后的走势分类---walkspeed
			
同级别走势的中枢震荡有重叠,即意味当下级别走势类型是不能延续啦.走势级别开始升级.根据走势分解定理,可知走势能划分出至少三段当下级别的走势类型. 有三段同级别完成的走势类型,就必须有三个同级别的中枢. ...
 - 微信video和audio无法自动播放解决方案
			
//音频,写法一<audio src="music/bg.mp3" autoplay loop controls>你的浏览器还不支持哦</audio> // ...
 - CSS样式命名整理
			
CSS样式命名整理 页面结构 容器: container/wrap 整体宽度:wrapper 页头:header 内容:content 页面主体:main 页尾:footer 导航:nav 侧栏:si ...
 - LoadRunner监控图表与配置(二)监控运行状况和交易状况
			
1.在左侧Available Graphs视图中展开Runtime Graphs节点,选择其中一种类型添加至控制器运行标签的界面. 2.在图中显示的空白区域点击右键,在弹出的快捷菜单中选择config ...
 - the art of seo(chapter ten)
			
Mobile, Local, and Vertical SEO ***The Mobile Landscape***Mobile site speed:• Google Page Speed Insi ...
 - openfire性能测试
			
使用TSung对Jabber服务器openfire进行压力测试 http://blog.csdn.net/spider_zhcl/article/details/6073920 Tsung负载测试Ti ...
 - bzoj 3267: KC采花&&3272&&3638&&3502 线段树
			
题目大意 给定一个长为n的序列,维护两种操作: 1.单点修改 2.在[l,r]这段区间中取k个互不相交的子段,使子段之和最大. \(n \leq 50000,k \leq 20\) 题解 四倍经验.( ...