zoukankan      html  css  js  c++  java
  • 基于Vue、Springboot网站实现第三方登录之QQ登录,以及邮件发送


    前言

    今天将基于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登录

    四、结束~

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

    辰鬼丫
  • 相关阅读:
    用Python抓取并分析了1982场英雄联盟数据,教你开局前预测游戏对局胜负!
    初级练手项目——用Python一步一步实现“智能”贪吃蛇!
    这五本Python急速入门必读的书,送给正在学习Python的你!
    探讨2018年最受欢迎的15顶级Python库!
    运用Python制作你心目中的完美女神脸!
    掌握这些Python代码技巧,编程至少快一半!
    解决Protege使用中的一个问题:从已有owl文件建立项目失败(w3china)
    http响应状态码
    云计算二 转帖
    protege中Property Domains and Ranges理解
  • 原文地址:https://www.cnblogs.com/Acechengui/p/13635460.html
Copyright © 2011-2022 走看看