前言

今天将基于Vue、Springboot实现第三方登录之QQ登录,以及QQ邮件发送给完成了,分享出来给大家~


提示:以下是本篇文章正文内容,下面案例可供参考

一、前提(准备)

要实现QQ登录这个功能呢,首先你得拥有一个备案好了的域名,然后前往QQ互联注册成为开发者,

传送门QQ互联

审核通过之后呢,按照提示,创建一个网站应用,



注意PS:上图中的网站地址信息务必要和你的网站对应的信息相符合,否则是无法审核通过的哦~



创建完成后等待审核通过,比如我的:

进行下一步~~

要实现QQ邮件发送呢,首先登录你的QQ邮箱,这里是传送门:

QQ邮箱登录

进入QQ邮箱页面,点击设置,选择账户,如下图:



准备工作完成!

二、QQ登录实现

1.前端

这里只贴相关代码,页面部分如下:

	  <!-- 第三方登录 -->
<el-divider>第三方登录</el-divider>
<div>
<a class="icon-qq" @click="LoginClick('qq')" href="javascript:void(0);">
<svg class="icon" aria-hidden="true">
<use xlink:href="#icon-QQ"></use></svg></a>
</div>

methods中的代码如下:

	 // 第三方登录
LoginClick(value) {
if (value == "qq") {
var _this = this;
this.$axios.get('/qq/oauth').then(resp =>{
//window.open(resp.data.result, "_blank")
var width = width || 900;
var height = height || 540;
var left = (window.screen.width - width) / 2;
var top = (window.screen.height - height) / 2;
var win =window.open(resp.data.result, "_blank",
"toolbar=yes, location=yes, directories=no, status=no, menubar=yes, scrollbars=yes, resizable=no, copyhistory=yes, left=" +
left + ", top=" + top + ", width=" + width + ", height=" + height);
//监听登录窗口是否关闭,登录成功后 后端返回关闭窗口的代码
var listener=setInterval(function(){
//监听窗口是否关闭
if(win.closed){
//进入这个if代表后端验证成功!直接跳转路由
clearInterval(listener);//关闭监听
//跳转路由
var path = _this.$route.query.redirect;
_this.$router.replace({
path:
path === "/" || path === undefined ? "/admin/dashboard" : path
});
_this.$router.go(0) //刷新
}
},500) }).catch(fail =>{
console.error(fail)
})
}
},

上述代码中,由 @click=“LoginClick(‘qq’)” 触发事件,像后端发送请求,响应成功后,获取响应回来的数据 resp.data.result 通过window.open()方法,在当前页面再打开一个窗口。效果大致如下:



为什么能显示这样效果呢?下面我们来看后端springboot代码~~

2.后端

导入所需依赖:

<!-- QQ第三方登录所需 -->
<!--httpclient-->
<dependency>
<groupId>org.apache.httpcomponents</groupId>
<artifactId>httpclient</artifactId>
<version> 4.5.9 </version>
</dependency>
<dependency>
<groupId>net.gplatform</groupId>
<artifactId>Sdk4J</artifactId>
<version>2.0</version>
</dependency>

1.application.yml 和工具类QQHttpClient

代码如下:

qq:
oauth:
http: http://www.wangxingjun.top/ #QQ互联中填写的网站地址

下面编写工具类 QQHttpClient,注意看注释,完整代码如下:

package top.wangxingjun.separate.util;

import com.alibaba.fastjson.JSONObject;
import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.util.EntityUtils; import java.io.IOException; /**
* @author wxj
* QQ工具类(主要用于解析QQ返回的信息)
*/
public class QQHttpClient {
//QQ互联中提供的 appid 和 appkey
public static final String APPID = "qq互联创建的应用AppID"; public static final String APPKEY = "qq互联创建的应用Appkey"; private static JSONObject parseJSONP(String jsonp){
int startIndex = jsonp.indexOf("(");
int endIndex = jsonp.lastIndexOf(")"); String json = jsonp.substring(startIndex + 1,endIndex); return JSONObject.parseObject(json);
}
//qq返回信息:access_token=FE04************************CCE2&expires_in=7776000&refresh_token=88E4************************BE14
public static String getAccessToken(String url) throws IOException {
CloseableHttpClient client = HttpClients.createDefault();
String token = null; HttpGet httpGet = new HttpGet(url);
HttpResponse response = client.execute(httpGet);
HttpEntity entity = response.getEntity(); if(entity != null){
String result = EntityUtils.toString(entity,"UTF-8");
if(result.indexOf("access_token") >= 0){
String[] array = result.split("&");
for (String str : array){
if(str.indexOf("access_token") >= 0){
token = str.substring(str.indexOf("=") + 1);
break;
}
}
}
} httpGet.releaseConnection();
return token;
}
//qq返回信息:callback( {"client_id":"YOUR_APPID","openid":"YOUR_OPENID"} ); 需要用到上面自己定义的解析方法parseJSONP
public static String getOpenID(String url) throws IOException {
JSONObject jsonObject = null;
CloseableHttpClient client = HttpClients.createDefault(); HttpGet httpGet = new HttpGet(url);
HttpResponse response = client.execute(httpGet);
HttpEntity entity = response.getEntity(); if(entity != null){
String result = EntityUtils.toString(entity,"UTF-8");
jsonObject = parseJSONP(result);
} httpGet.releaseConnection(); if(jsonObject != null){
return jsonObject.getString("openid");
}else {
return null;
}
} //qq返回信息:{ "ret":0, "msg":"", "nickname":"YOUR_NICK_NAME", ... },为JSON格式,直接使用JSONObject对象解析
public static JSONObject getUserInfo(String url) throws IOException {
JSONObject jsonObject = null;
CloseableHttpClient client = HttpClients.createDefault(); HttpGet httpGet = new HttpGet(url);
HttpResponse response = client.execute(httpGet);
HttpEntity entity = response.getEntity(); if(entity != null){
String result = EntityUtils.toString(entity,"UTF-8");
jsonObject = JSONObject.parseObject(result);
} httpGet.releaseConnection(); return jsonObject;
}
}

2.QQLoginController

下面是控制层完整代码,如下:

package top.wangxingjun.separate.controller;

import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;
import lombok.extern.log4j.Log4j2;
import org.apache.shiro.SecurityUtils;
import org.apache.shiro.authc.UsernamePasswordToken;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.ResponseBody;
import top.wangxingjun.separate.entity.User;
import top.wangxingjun.separate.result.Result;
import top.wangxingjun.separate.result.ResultFactory;
import top.wangxingjun.separate.service.UserService;
import top.wangxingjun.separate.util.QQHttpClient; import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.net.URLEncoder;
import java.util.UUID; /**
* @author wxj
* @create 2019-09-27 20:32
* QQ第三方登陆接口
*/
@Controller
@Log4j2
public class QQLoginController{
@Value("${qq.oauth.http}")
private String http; @Autowired
private UserService userService; /**
* 发起请求
* @param session
* @return
*/
@GetMapping("/api/qq/oauth")
@ResponseBody
public Result qq(HttpSession session){
//QQ互联中的回调地址
String backUrl = http + "api/qq/callback"; //用于第三方应用防止CSRF攻击
String uuid = UUID.randomUUID().toString().replaceAll("-","");
session.setAttribute("state",uuid); //Step1:获取Authorization Code
String url = "https://graph.qq.com/oauth2.0/authorize?response_type=code"+
"&client_id=" + QQHttpClient.APPID +
"&redirect_uri=" + URLEncoder.encode(backUrl) +
"&state=" + uuid;
return ResultFactory.buildSuccessResult(url);
} /**
* QQ回调 注意 @GetMapping("/qq/callback")路径
* 是要与QQ互联填写的回调路径一致(我这里因为前端请求愿意不用写成 api/qq/callback)
* @param request
* @return
*/
@GetMapping("/qq/callback")
@ResponseBody
public String qqcallback(HttpServletRequest request,HttpServletResponse response) throws Exception {
response.setContentType("text/html; charset=utf-8"); // 响应编码
HttpSession session = request.getSession();
//qq返回的信息:http://graph.qq.com/demo/index.jsp?code=9A5F************************06AF&state=test
String code = request.getParameter("code");
String state = request.getParameter("state");
String uuid = (String) session.getAttribute("state"); if(uuid != null){
if(!uuid.equals(state)){
throw new Exception("QQ,state错误");
}
}
//Step2:通过Authorization Code获取Access Token
String backUrl = http + "/qq/callback";
String url = "https://graph.qq.com/oauth2.0/token?grant_type=authorization_code"+
"&client_id=" + QQHttpClient.APPID +
"&client_secret=" + QQHttpClient.APPKEY +
"&code=" + code +
"&redirect_uri=" + backUrl; String access_token = QQHttpClient.getAccessToken(url); //Step3: 获取回调后的 openid 值
url = "https://graph.qq.com/oauth2.0/me?access_token=" + access_token;
String openid = QQHttpClient.getOpenID(url); //Step4:获取QQ用户信息
url = "https://graph.qq.com/user/get_user_info?access_token=" + access_token +
"&oauth_consumer_key="+ QQHttpClient.APPID +
"&openid=" + openid; JSONObject jsonObject = QQHttpClient.getUserInfo(url);
//可以放到Redis和mysql中
//session.setAttribute("openid",openid); //openid,用来唯一标识qq用户
//session.setAttribute("nickname",(String)jsonObject.get("nickname")); //QQ名
//session.setAttribute("figureurl_qq_2",(String)jsonObject.get("figureurl_qq_2")); //大小为100*100像素的QQ头像URL
if(!userService.qqisExist(openid)){//用户不存在
User u=new User();
u.setUsername(openid);
u.setPassword("123456");
u.setOpenid(openid);
u.setName((String)jsonObject.get("nickname"));
u.setTpath((String)jsonObject.get("figureurl_qq_2"));
u.setEnabled(true);
//注册
userService.qqregister(u);
//redirect:../admin/dashboard
return "<script>window.close();</script>";
}
User us=userService.findByOpenid(openid);
UsernamePasswordToken token = new UsernamePasswordToken(us.getUsername(), "123456");
SecurityUtils.getSubject().login(token);
return "<script>window.opener.localStorage.setItem('username', '"+JSON.toJSONString(us)+"');window.close();</script>";
}
}

我们可以看到,控制层中有两处方法,第一个方法就是发起登录的请求处理方法,第二个方法就是处理前端打开了QQ授权窗口之后,授权成功的回调请求处理方法(注意路径要和你在QQ互联上填写回调路径一致)。第二个方法里面便可以处理你的具体逻辑,比如是否是第一次登录,按照自己业务需求来~ 而我这里的处理逻辑是:第一次登录,那就进行注册,然后通过返回一段js代码,使前台的qq窗口关闭,这里我们重点要注意,看图:



对应的前端代码,如下图:



通过监听窗口是否关闭,来进行下一步逻辑处理。


三、邮件发送

上面已经将qq登录相关的介绍完了,下面将邮件发送也补充上!(前期准备写在上面了)

我这里的处理逻辑是:qq登录时,如果从数据库中通过openid查询数据,不存在即表示需要注册,然后调用注册方法注册。如果数据库存在,即直接通过openid查询对应信息,然后通过拼接js代码的方式,将其对应信息存入前端vue中的localStorage里面。看下面图:



图中的window.opener.localStorage.setItem(‘username’, ‘"+JSON.toJSONString(us)+"’);换成你自己的存放方法即可。重点是 window.close();能够将前端qq窗口关闭。而window.opener,指向的是当前父级浏览器窗口~

呃呃呃呃呃~咋又扯到qq登录相关的去了,算了,就算是解释补充吧,毕竟是结合使用的。

好了,继续下面的逻辑:当第一次登录时,进行注册,注册成功之后发送qq邮件给管理员,这样管理员就可以收到邮件。

下面看实现步骤:

1.引入依赖

<!--邮箱发送-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-mail</artifactId>
</dependency>

2.配置文件application.properties

# 邮件发送配置
# 因为是QQ邮箱,所以host需要使用smtp.qq.com。如果是其它邮箱,搜索下即可找到。
spring.mail.host=smtp.qq.com
# 这里便是写你在qq邮箱设置的@foxmail.com
spring.mail.username=acechengui@foxmail.com
# 这里便是写你在qq邮箱设置POP3/SMTP服务生成的令牌
spring.mail.password=XXXXXXXXXX
# 编码格式
spring.mail.default-encoding=UTF-8
#开启加密验证
spring.mail.properties.mail.smtp.ssl.enable=true

3.编码

 @Autowired
private JavaMailSender javaMailSender;

注入之后,我这里是将邮件发送,写在业务层,相关代码如下:

/**
* 邮件发送
* @param username
* @throws MessagingException
* From需要和配置文件中的username一致,否则会报错。
* To为邮件接收者;
* Subject为邮件的标题;
* Text为邮件的内容。
*/
public void mailSend(String username) throws MessagingException {
//发送简单邮件
//SimpleMailMessage message = new SimpleMailMessage();
//message.setFrom("acechengui@foxmail.com");
//message.setTo("623169670@qq.com");
//message.setSubject("用户注册通知");
//message.setText("用户"+username+"注册成功,请及时进行赋权");
//javaMailSender.send(message); //发送HTML邮件
MimeMessage mimeMessage = javaMailSender.createMimeMessage();
MimeMessageHelper messageHelper = new MimeMessageHelper(mimeMessage);
messageHelper.setSubject("用户注册通知");
messageHelper.setFrom("acechengui@foxmail.com");
messageHelper.setTo("623169670@qq.com");
messageHelper.setText("<a href='javascript:void(0)'>用户"+username+"注册成功,请及时进行赋权</a>", true);
javaMailSender.send(messageHelper.getMimeMessage());
}

再到需要发送的地方调用此方法,这里有两种发送邮件方式,一个简单发送,另一种自定义格式的发送,当然还有携带文件的发送,那就的自行百度了实现起来基本一样,比如我这里是注册后发送邮件,看如下图调用即可:



到此,邮件发送使用介绍结束了,下面来看我的效果,看视频:

vue springboot实现qq邮件发送

下面来看下注册成功之后再进行qq登录,看视频:

vue springboot实现qq登录

四、结束~

自己不自觉不够努力,就不要怪别人不管你,不提醒你~ ------------------辰鬼

基于Vue、Springboot网站实现第三方登录之QQ登录,以及邮件发送的更多相关文章

  1. 网站如何接入第三方登录,微信登录和QQ登录:注册认证篇

    第三方登录平台接入 (QQ\微信登录) QQ登录接入 第一步成为QQ应用开发者,审核期限七天 一.所需材料 1.公司注册相关信息 2.营业执照扫描件 微信登录接入 第一步成为微信开发平台开发者,认证费 ...

  2. 第三方登录:QQ登录实现(OAuth2.0)

    一.创建应用 1.在 QQ互联 创建应用 地址:https://connect.qq.com/manage.html#/ 然后进行实名认证,创建应用,审核通过 然后点击查看,可以获得 APP ID 和 ...

  3. 从零玩转第三方登录之QQ登录

    从零玩转第三方登录之QQ登录 前言 在真正开始对接之前,我们先来聊一聊后台的方案设计.既然是对接第三方登录,那就免不了如何将用户信息保存.首先需要明确一点的是,用户在第三方登录成功之后, 我们能拿到的 ...

  4. 第三方登录(QQ登录)开发流程详解

    原文:http://www.cnblogs.com/it-cen/p/4338202.html 近排由于工作的繁忙,已经一个星期没写博文做分享了,接下来我对网站接入第三方登录----QQ登录的实现逻辑 ...

  5. (转)第三方登录(QQ登录)开发流程详解

    近排由于工作的繁忙,已经一个星期没写博文做分享了,接下来我对网站接入第三方登录----QQ登录的实现逻辑做一个详细的讲解. 对于整个流程的详细文档可以到QQ互联官网(http://wiki.conne ...

  6. dedecms织梦第三方登录插件-QQ登录、微博登录、微信登录

    织梦程序集成第三方QQ登录.微博登录.微信登录,获取QQ.微博.微信,并存储至数据库,一键注册为网站会员,不用再次填写绑定信息,方便粘贴用户更强. 织梦第三方登录效果 第三方登录插件特点 1.所有文件 ...

  7. Android之第三方平台实现QQ登录和QQ分享

    目前大多数APP都包含了第三方平台的登录,特别是QQ和微信,这篇博客主要讲的是如何实现QQ第三方平台实现QQ登录和分享功能,功能包含: 登录授权登录获取用户信息(昵称,头像,地址等) QQ分享给好友 ...

  8. Spring Security构建Rest服务-0102-Spring Social开发第三方登录之qq登录

    图一 基于SpringSocial实现qq登录,要走一个OAuth流程,拿到服务提供商qq返回的用户信息. 由上篇介绍的可知,用户信息被封装在了Connection里,所以最终要拿到Connectio ...

  9. PHP 接入(第三方登录)QQ 登录 OAuth2.0 过程中遇到的坑

    前言 绝大多数网站都集成了第三方登录,降低了注册门槛,增强了用户体验.最近看了看 QQ 互联上 QQ 登录的接口文档.接入 QQ 登录的一般流程是这样的:先申请开发者 -> 然后创建应用(拿到一 ...

随机推荐

  1. Parcelable使用(二)

    简单的Activity间数据传递用Intent,复杂的用Parcelable,举个栗子第一个activity写入Parcel的String类型的name和int类型的age,第二个activity取出 ...

  2. 利用云主机搭建MySQL服务器

    前言:有了一台云服务器之后,就想着如何物尽其用.利用其不关机(意外除外)的特性,我们可以在服务器上安装数据库服务,实现云数据库服务器,这样就可以随时随地的访问数据库了,不再受各种限制. 这里以MySQ ...

  3. Pytest 系列(25)- 标记用例级别 @allure.

    如果你还想从头学起Pytest,可以看看这个系列的文章哦! https://www.cnblogs.com/poloyy/category/1690628.html 前言 平时写测试用例也会划分优先级 ...

  4. Redis事务使用方法

    Redis事务 Redis事务是一组命令的集合,也是Redis的最小执行单位之一.一个事务的所有命令,要么都执行,要么都不执行.Redis能保证事务执行期间不会有其他命令插入. 相关命令 命令 格式 ...

  5. Java学习的第十四天

    1.JAVA的继承使用关键字extends继承 class 子类 extends 父类{} 子类可以用super来调用父类的非私有属性和非私有方法,还可以调用非私有的父类构造函数 如果父类子类的方法名 ...

  6. 仅用六种字符来完成Hello World,你能做到吗?

    Hello World 对于每一个开发者来说都不陌生,因为在我们学习任何一个语言或框架的时候,都会有一个Hello World的案例来帮助我们快速入门. 如果我们使用JavaScript来输出Hell ...

  7. 微信小程序获取高宽uniapp

    代码片段 <template> <view> <view class="text" id="w">补充文字</view ...

  8. (二)http请求方法和状态码

    1.HTTP请求方法 根据 HTTP 标准,HTTP 请求可以使用多种请求方法. HTTP1.0 定义了三种请求方法: GET.POST 和 HEAD方法. HTTP1.1 新增了六种请求方法:OPT ...

  9. 【Java】类的结构

    类与对象 类中主要包括五种结构,下面进行对这五种结构进行详细的介绍. 1. 面向对象与面向过程 面向过程:强调的是功能行为,以函数为最小单位,考虑怎么做. 面向对象:强调具备了功能的对象,以类/对象为 ...

  10. CF1295E Permutation Separation

    线段树 难得把E想出来,写出来,但却没有调出来(再给我5分钟),我的紫名啊,我一场上紫的大好机会啊 首先考虑是否能将$k$在$1$--$n-1$的每一个的最小代价都求出来 因为$k$从$i$到$i-1 ...