Spring Boot集成第三方登錄之微博登錄

準(zhǔn)備工作

網(wǎng)站接入

登陸微博開放平臺座享,進(jìn)入微連接绍绘,選擇網(wǎng)站接入
image.png

點(diǎn)擊立即接入
image.png

開發(fā)者信息認(rèn)證

填寫開發(fā)者信息與身份認(rèn)證信息
image.png

創(chuàng)建應(yīng)用

開發(fā)者信息認(rèn)證通過后即可創(chuàng)建應(yīng)用凌外。
image.png

應(yīng)用創(chuàng)建成功后會得到app keyapp secret

image.png

在應(yīng)用信息的高級信息中設(shè)置授權(quán)回調(diào)地址
image.png

添加測試賬號佑女,這里使用開發(fā)者賬號測試奥秆。
image.png

流程分析

Web網(wǎng)站的授權(quán)流程如下
image.png

引導(dǎo)授權(quán)用戶

引導(dǎo)需要授權(quán)的用戶到如下地址

https://api.weibo.com/oauth2/authorize?client_id=YOUR_CLIENT_ID&response_type=code&redirect_uri=YOUR_REGISTERED_REDIRECT_URI
 <a >
    <span>微博</span>
</a>
image.png

用戶授權(quán)

來到授權(quán)地址后歉备,需要用戶授權(quán)常摧,授權(quán)成功返回Code碼

image.png

授權(quán)成功

如果用戶同意授權(quán),頁面跳轉(zhuǎn)至回調(diào)地址并攜帶Code

image.png

換取Access Token

通過授權(quán)返回的CODE威创,請求如下地址落午,換取Access Token。

https://api.weibo.com/oauth2/access_token?client_id=YOUR_CLIENT_ID&client_secret=YOUR_CLIENT_SECRET&grant_type=authorization_code&redirect_uri=YOUR_REGISTERED_REDIRECT_URI&code=CODE
https://api.weibo.com/oauth2/access_token?code=0c55753fe19a5bcf0a4a42afd4a64353&grant_type=authorization_code&client_secret=81abcdefghijkmllb4a45f4288ef&redirect_uri=https://ws20264753.zicp.fun/weibo/success&client_id=231234566

得到如下響應(yīng)結(jié)果

body: {"access_token":"2.00123456789480Ngwx8","remind_in":"15896999","expires_in":157679999,"uid":"12345678","isRealName":"true"}

HTTP客戶端
Code獲取后肚豺,需要使用Code獲取Access Token溃斋,以及使用token請求其他接口,此時需要使用發(fā)送請求的HTTP客戶端吸申,這里使用hutool工具類

發(fā)送一個POST請求示例:

//鏈?zhǔn)綐?gòu)建請求
String result2 = HttpRequest.post(url)
    .header(Header.USER_AGENT, "Hutool http")//頭信息梗劫,多個頭信息多次調(diào)用此方法即可
    .form(paramMap)//表單內(nèi)容
    .timeout(20000)//超時,毫秒
    .execute().body();
Console.log(result2);

使用Access Token請求相關(guān)接口

Access Token得到后截碴,就可以使用Access Token請求相關(guān)接口獲取對應(yīng)數(shù)據(jù)梳侨,這里使用2個接口舉例使用說明。

相關(guān)接口如下:

image.png

根據(jù)用戶ID獲取用戶信息

根據(jù)用戶ID獲取用戶信息接口描述信息如下
image.png

只需要如下設(shè)置請求參數(shù)日丹,請求地址即可獲取用戶信息

Map<String, Object> selectUserParam = new HashMap<>();
selectUserParam.put("access_token", "token");
selectUserParam.put("uid", "uid");

HttpResponse execute = HttpRequest.get("https://api.weibo.com/2/users/show.json").form(selectUserParam).timeout(2000).execute();

響應(yīng)如下類似信息

{
    "id": 51234100,
    "idstr": "58812345464100",
    "class": 1,
    "screen_name": "XX",
    "name": "XX",
    "province": "51",
    "city": "14",
    "location": "四川 成都",
    "description": "真正的強(qiáng)者走哺,不是流淚的人,而是含淚奔跑的人哲虾。",
    "url": "",
    "profile_image_url": "https://tva3.sinaimg.cn/crop.0.0.996.996.50/006qils8jw8f2cztnnp6vj30ro0rpgnj.jpg?KID=imgbed,tva&Expires=1663398489&ssig=JM0ZTUEYbs",
    "light_ring": false,
    "cover_image_phone": "http://ww1.sinaimg.cn/crop.0.0.640.640.640/549d0121tw1egm1kjly3jj20hs0hsq4f.jpg",
    "profile_url": "u/58812345464100",
    "domain": "",
    "weihao": "",
    "gender": "m",
}

獲取用戶的粉絲列表

獲取用戶的粉絲列表接口描述信息如下
image.png
Map<String, Object> selectUserParam = new HashMap<>();
selectUserParam.put("access_token", weiboUser.getAccess_token());
selectUserParam.put("uid", weiboUser.getUid());

HttpResponse execute = HttpRequest.get("https://api.weibo.com/2/friendships/followers.json").form(selectUserParam).timeout(2000).execute();

響應(yīng)結(jié)果

粉絲信息: {"users":[],"has_filtered_attentions":false,"next_cursor":600,"previous_cursor":0,"total_number":1789,"use_sink_stragety":false,"has_filtered_fans":true,"use_status_strategy":false,"show_related_topic":false,"display_total_number":1789,"display_total_number_str":"1789","show_unread":false}

Spring Boot集成微博登錄

添加依賴

添加hutool開發(fā)工具包丙躏,核心用于發(fā)送Http請求择示。

<dependency>
     <groupId>cn.hutool</groupId>
     <artifactId>hutool-all</artifactId>
     <version>5.8.2</version>
</dependency>

<!-- thymeleaf頁面 -->
<dependency>
   <groupId>org.springframework.boot</groupId>
   <artifactId>spring-boot-starter-thymeleaf</artifactId>
</dependency>

封裝Token等信息

用戶授權(quán)成功后,使用WeiboUser類封裝相關(guān)信息

/**
 * 封裝登錄認(rèn)證后的令牌等信息
 */
@Data
public class WeiboUser {
    /**
     * 令牌
     */
    private String access_token;
    /**
     * 令牌過期時間,該參數(shù)即將廢棄
     */
    private String remind_in;
    /**
     * 令牌過期時間,單位是秒數(shù)
     */
    private long expires_in;
    /**
     * 該社交用戶的唯一標(biāo)識
     */
    private String uid;
    /**
     * 是否記住我
     */
    private String isRealName;
}

創(chuàng)建Login頁面

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<div>
    <a >
        <span>微博</span>
    </a>
</div>
</body>
</html>

創(chuàng)建Home頁面

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
  <div>
    <h3>登錄成功</h3>
  </div>
</body>
</html>

微博登錄邏輯

package com.example.demo.controller;

import cn.hutool.http.HttpRequest;
import cn.hutool.http.HttpResponse;
import com.alibaba.fastjson.JSON;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;

import javax.servlet.http.HttpSession;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;
import java.util.HashMap;
import java.util.Map;

@Slf4j
@Controller
public class WeiBoController {


    @RequestMapping("/weibo/login")
    public String login() {
        return "login";
    }

    @RequestMapping("/weibo/success")
    public String authorize(String code, HttpSession session) throws Exception {
        // 使用code換取token晒旅,換取成功則繼續(xù)栅盲,否則重定向登錄頁
        Map<String, Object> paramMap = new HashMap<>();
        paramMap.put("client_id", "2312345676");
        paramMap.put("client_secret", "819b8cd2dbd1f188dbedb4a45f4288ef");
        paramMap.put("grant_type", "authorization_code");
        paramMap.put("redirect_uri", "https://ws20264753.zicp.fun/weibo/success");
        paramMap.put("code", code);
        String url = this.buildUrl("https://api.weibo.com", "/oauth2/access_token", paramMap);
        //發(fā)送post請求換取token
        HttpResponse httpResponse = HttpRequest.post(url)
                .timeout(20000)//超時,毫秒
                .execute();

        Map<String, String> errors = new HashMap<>();
        if (httpResponse.getStatus() == 200) {
            String body = httpResponse.body();
            log.info("body: {}", body);
            WeiboUser weiboUser = JSON.parseObject(body, WeiboUser.class);
            // TODO 使用 weiboUser.getUid() 查詢數(shù)據(jù)庫 若查詢結(jié)果為null废恋,則之前未登陸過谈秫,查詢其社交信息進(jìn)行注冊
            Boolean selectUser = true;
            if (selectUser) {
                log.info("用戶未注冊,查詢用戶信息進(jìn)行注冊");
                this.register(weiboUser);

                // 獲取粉絲信息
                this.getFan(weiboUser);
            } else {
                log.info("用戶已注冊鱼鼓,更新相關(guān)信息");
                // TODO 更新TOKEN拟烫、UID、登錄過期時間等信息
                weiboUser.getAccess_token();
                weiboUser.getUid();
                weiboUser.getExpires_in();
            }

            // 將用戶信息返回
            session.setAttribute("userInfo", weiboUser.toString());
            return "home";
        } else {
            errors.put("msg", "獲得第三方授權(quán)失敗蚓哩,請重試");
            session.setAttribute("errors", errors);
            return "login";
        }
    }


    @RequestMapping("/weibo/fail")
    public void authorize() {
        log.info("weibo fail...");
    }


    /**
     * 獲取用戶信息
     */
    public void register(WeiboUser weiboUser) {
        Map<String, Object> selectUserParam = new HashMap<>();
        selectUserParam.put("access_token", weiboUser.getAccess_token());
        selectUserParam.put("uid", weiboUser.getUid());
//      String selectUserUrl = this.buildUrl("https://api.weibo.com", "/2/users/show.json", selectUserParam);

        HttpResponse execute = HttpRequest.get("https://api.weibo.com/2/users/show.json").form(selectUserParam).timeout(2000).execute();
        if (execute.getStatus() == 200) {
            String userInfo = execute.body();
            log.info("userInfo: {}", userInfo);
            //TODO 調(diào)用微博api接口獲取用戶信息,然后進(jìn)行注冊,記錄以下值
            weiboUser.getAccess_token();
            weiboUser.getUid();
            weiboUser.getExpires_in();
        }
    }

    /**
     * 獲取
     */
    public void getFan(WeiboUser weiboUser) {
        Map<String, Object> selectUserParam = new HashMap<>();
        selectUserParam.put("access_token", weiboUser.getAccess_token());
        selectUserParam.put("uid", weiboUser.getUid());
//      String selectUserUrl = this.buildUrl("https://api.weibo.com", "/2/users/show.json", selectUserParam);

        HttpResponse execute = HttpRequest.get("https://api.weibo.com/2/friendships/followers.json").form(selectUserParam).timeout(2000).execute();
        if (execute.getStatus() == 200) {
            String fanList = execute.body();
            log.info("粉絲信息: {}", fanList);
        }
    }


    /**
     * 構(gòu)建請求URl地址
     *
     * @return
     */
    private static String buildUrl(String host, String path, Map<String, Object> querys) throws
            UnsupportedEncodingException {
        StringBuilder sbUrl = new StringBuilder();
        sbUrl.append(host).append(path);

        StringBuilder sbQuery = new StringBuilder();

        for (Map.Entry<String, Object> query : querys.entrySet()) {
            if (sbQuery.length() > 0) {
                sbQuery.append("&");
            }
            sbQuery.append(query.getKey());
            sbQuery.append("=");
            sbQuery.append(URLEncoder.encode(query.getValue().toString(), "utf-8"));
        }

        if (sbQuery.length() > 0) {
            sbUrl.append("?").append(sbQuery);
        }

        return sbUrl.toString();
    }
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末上渴,一起剝皮案震驚了整個濱河市岸梨,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌稠氮,老刑警劉巖曹阔,帶你破解...
    沈念sama閱讀 218,682評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異隔披,居然都是意外死亡赃份,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,277評論 3 395
  • 文/潘曉璐 我一進(jìn)店門奢米,熙熙樓的掌柜王于貴愁眉苦臉地迎上來抓韩,“玉大人,你說我怎么就攤上這事鬓长≮怂” “怎么了?”我有些...
    開封第一講書人閱讀 165,083評論 0 355
  • 文/不壞的土叔 我叫張陵涉波,是天一觀的道長英上。 經(jīng)常有香客問我,道長啤覆,這世上最難降的妖魔是什么苍日? 我笑而不...
    開封第一講書人閱讀 58,763評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮窗声,結(jié)果婚禮上相恃,老公的妹妹穿的比我還像新娘。我一直安慰自己笨觅,他們只是感情好豆茫,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,785評論 6 392
  • 文/花漫 我一把揭開白布侨歉。 她就那樣靜靜地躺著,像睡著了一般揩魂。 火紅的嫁衣襯著肌膚如雪幽邓。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,624評論 1 305
  • 那天火脉,我揣著相機(jī)與錄音牵舵,去河邊找鬼。 笑死倦挂,一個胖子當(dāng)著我的面吹牛畸颅,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播方援,決...
    沈念sama閱讀 40,358評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼没炒,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了犯戏?” 一聲冷哼從身側(cè)響起送火,我...
    開封第一講書人閱讀 39,261評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎先匪,沒想到半個月后种吸,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,722評論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡呀非,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,900評論 3 336
  • 正文 我和宋清朗相戀三年坚俗,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片岸裙。...
    茶點(diǎn)故事閱讀 40,030評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡猖败,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出降允,到底是詐尸還是另有隱情辙浑,我是刑警寧澤,帶...
    沈念sama閱讀 35,737評論 5 346
  • 正文 年R本政府宣布拟糕,位于F島的核電站判呕,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏送滞。R本人自食惡果不足惜侠草,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,360評論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望犁嗅。 院中可真熱鬧边涕,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,941評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至式撼,卻和暖如春童社,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背著隆。 一陣腳步聲響...
    開封第一講書人閱讀 33,057評論 1 270
  • 我被黑心中介騙來泰國打工扰楼, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人美浦。 一個月前我還...
    沈念sama閱讀 48,237評論 3 371
  • 正文 我出身青樓弦赖,卻偏偏與公主長得像,于是被迫代替她去往敵國和親浦辨。 傳聞我的和親對象是個殘疾皇子蹬竖,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,976評論 2 355

推薦閱讀更多精彩內(nèi)容