Python实现新版正方系统滑动验证码识别
作者:狂野小青年 发布时间:2022-11-08 09:14:32
标签:Python,滑动验证码,识别
Python实现新版正方系统滑动验证码识别算法和方案
步骤一:点击数据分析
点击滑动按钮,将发送一个请求到 /zfcaptchaLogin
请求内容
"type": "verify"
"rtk": "6cfab177-afb2-434e-bacf-06840c12e7af"
"time": "1624611806948"
"mt": "W3sieCI6OTY1LCJ5IjoxNjksInQiOjE2MjQ2MTE4MDY4Njh9LHsieCI6OTY1LCJ5IjoxNjksInQiOjE2MjQ2MTE4MDY5NDh9XQ=="
"instanceId": "zfcaptchaLogin"
"extend": "eyJhcHBOYW1lIjoiTmV0c2NhcGUiLCJ1c2VyQWdlbnQiOiJNb3ppbGxhLzUuMCAoTWFjaW50b3NoOyBJbnRlbCBNYWMgT1MgWCAxMF8xNV83KSBBcHBsZVdlYktpdC81MzcuMzYgKEtIVE1MLCBsaWtlIEdlY2tvKSBDaHJvbWUvOTEuMC40NDcyLjEwNiBTYWZhcmkvNTM3LjM2IiwiYXBwVmVyc2lvbiI6IjUuMCAoTWFjaW50b3NoOyBJbnRlbCBNYWMgT1MgWCAxMF8xNV83KSBBcHBsZVdlYktpdC81MzcuMzYgKEtIVE1MLCBsaWtlIEdlY2tvKSBDaHJvbWUvOTEuMC40NDcyLjEwNiBTYWZhcmkvNTM3LjM2In0="
通过 base64 解密 mt和 extend 得出解密的数值
# mt
[{"x":965,"y":169,"t":1624611806868},{"x":965,"y":169,"t":1624611806948}]
# extend
{"appName":"Netscape","userAgent":"Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.106 Safari/537.36","appVersion":"5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.106 Safari/537.36"}
mt 为用户的点击行为,x为X轴上的值,y为Y轴上的值,t为时间戳。通过大量点击分析,发现x值最小值为 950,得出950 为 X轴的起点,y值随机无固定值。
extend 为请求头部内容
步骤二:滑动验证码图像分析,计算滑动距离x值
将图像灰度化,通过getpixel可以获取图像某一点的颜色值, 颜色值越高代表图像越浅,所以寻找纵向连续50个像素点均是 getpixel(x+1, y) > getpixel(x, y)(X轴=x 比 X轴=x+1 颜色浅)
并扫描图像,当x=130、扫描高度=50时,的颜色比x+1时深。
from PIL import Image
import matplotlib.pyplot as plt
import numpy as np
scanf_height= 50 # 扫描的高度
img = Image.open("zfcaptchaLogin.png")
def contrast(imgl, x, y,scanf_height):
# 黄框颜色值比红框颜色值浅的个数
count = 0
for i in range(scanf_height):
if imgl.getpixel((x+1, y+i)) > imgl.getpixel((x, y+i)):
count += 1
# 当 count = scanf_height, 代表黄条区域 整体 红条区域 颜色值浅,则是验证码框位置
return count
def scanf(img):
imgx, imgy = img.size
imgl = img.convert('L') # 图像灰度化
plt.yticks([])
plt.xticks([i for i in range(0, imgx, 25)])
plt.imshow(img)
plt.pause(0.5)
for y in range(0, imgy-scanf_height, 10):
plt.pause(0.01)
plt.clf()
plt.yticks([])
plt.xticks([i for i in range(0, imgx, 25)])
plt.imshow(imgl, cmap=plt.cm.gray)
for x in range(1, imgx-1, 1):
plt.pause(0.0001)
plt.plot([x-1,x-1], [y, y+scanf_height], color='white')
plt.plot([x,x], [y, y+scanf_height], color='red')
plt.plot([x+1,x+1], [y, y+scanf_height], color='yellow')
count = contrast(imgl, x,y, scanf_height)
plt.title('count: {}'.format(count) )
print("x,y=[{}, {}], 黄条区域值比红条区域颜色值浅的个数:{}".format(x,y, count))
if count == scanf_height:
return
scanf(img)
plt.show()
优化代码计算x,y值
import json
import random
import time
from io import BytesIO
from PIL import Image
class ZfCaptchaRecognit(object):
def __init__(self, img_path):
self.img = Image.open(img_path)
def _get_xy(self):
# 计算 x,y 值
def _is_dividing_line(img_l, x, y):
for n in range(50):
# 寻找纵向连续50个像素点均是 X=x 比 X=x+1 颜色深
if y + n >= img_l.size[1] or x >= img_l.size[0] - 1:
return False
if img_l.getpixel((x + 1, y + n)) - img_l.getpixel((x, y + n)) < 2:
return False
return True
img_l = self.img.convert("L")
for x in range(img_l.size[0]):
for y in range(img_l.size[1]):
if _is_dividing_line(img_l, x, y):
return (x, y)
def show_tag(self):
# 展示 切分点
X, Y = self._get_xy()
img2 = Image.new("RGB", self.img.size, (255, 255, 255))
for x in range(self.img.size[0]):
for y in range(self.img.size[1]):
pix = self.img.getpixel((x, y))
img2.putpixel((x, y), pix)
if x == X or y == Y:
img2.putpixel((x, y), 225)
img2.save("show_tag.png")
img2.show()
captcha = ZfCaptchaRecognit("zfcaptchaLogin.png")
captcha.show_tag()
步骤三:生成提交参数
通过 步骤一得出x值最小为950,y值无规律
则提交参数mt的大致格式数据是
[{
"x":950+ 滑动距离 + 浮动值, # 浮动值的范围通过分析提交参数得出在10~20内
"y":random.randint(150, 190), # 无规律,暂定150到190范围内
"t":int(time.time() * 1000)}, # 时间戳
...]
获取mt 参数
import json
import random
import time
from io import BytesIO
from PIL import Image
class ZfCaptchaRecognit(object):
def __init__(self, img_stream):
obj = BytesIO(img_stream)
self.img = Image.open(obj)
def _get_xy(self):
...
def generate_payload(self):
base_x = 950
X, Y = self._get_xy()
payloads = [{"x": base_x + random.randint(5, 20), "y": random.randint(150, 190), "t": int(time.time() * 1000)}]
for i in range(random.randint(15, 30)):
# 在上一个参数基础下浮动
last_payload = payloads[-1].copy()
payloads[0]["x"] += random.choice([0] * 8 + [1, -1] * 2 + [2, -2])
last_payload["t"] += random.randint(1, 20)
last_payload["y"] += random.choice([0] * 8 + [1, -1] * 2 + [2, -2])
payloads.append(last_payload)
payloads[-1]["x"] = base_x + random.randint(10, 20) + X
return json.dumps(payloads)
captcha = ZfCaptchaRecognit("zfcaptchaLogin.png")
captcha. generate_payload()
来源:https://blog.csdn.net/veloi/article/details/115416495
0
投稿
猜你喜欢
- 本文实例为大家分享了python实现五子棋游戏的具体代码,供大家参考,具体内容如下checkerboard.pyfrom collectio
- 1、$_SERVER$_SERVER超级全局变量包含由web服务器创建的信息,它提供了服务器和客户配置及当前请求环境的有关信息。根据服务器不
- ASP的强大不仅仅局限于接受和显示的交互,更多的是运用ActiveX 组件进行更强大的Web应用。那究竟ActiveX组件为何物?
- 继承与threading.Thread实现有返回值的子类MyThread,废话不多说,大家直接看代码import threadingclas
- 如果你细心跟踪一下SQL Server数据库服务器的登录过程,你会发现口令计算其实是非常脆弱的,SQL Server数据库的口令脆弱体现两方
- Access数据库,同时操作大量记录(9500条以上)时报错。错误提示:Microsoft JET Database Engine 错误 &
- 为了防止某些别有用心的人从外部访问数据库,盗取数据库中的用户姓名、密码、信用卡号等其他重要信息,在我们创建数据库驱动的解决方案时,我们首先需
- 以下介绍用数据库实现简单计数器,功能实现统计网站每日访问,每周访问及总访问量的统计,使用js调用下面存为count.asp<%&nbs
- 最近在无忧脚本混了一阵子,回复了一些贴子,自己却没有做出什么东东让大家看看,心里有些不安,于是写了下边的一点东西,本来应该发在类封装区的,考
- 适用环境: PHP5.2.x / mysql 5.0.xclass Mysql { priva
- 今天有个学生问我:页面中使用GIF格式,失真太大,怎么办呢?这个问题比较简单啊,只要用JPG就可以了。我们常用的页面的图片格式有三种,GIF
- 在用JS编写动画的时候,经常用会到布局转换,即在运动前将相对定位转为绝对定位,然后执行动画函数。下面给大家分享一个运用原生JS实现的布局转换
- 在默认情况下,MySQL搜索不区分大小写(但某些字符集始终区分大小写,如czech)。这意味着,如果你使用col_name LIKE
- php服务端与客户端交互、提供开放api时,通常需要对敏感的部分api数据传输进行数据加密,这时候rsa非对称加密就能派上用处了,下面通过一
- 概述今天我们要来做一个进阶的花分类问题. 不同于之前做过的鸢尾花, 这次我们会分析 102 中不同的花. 是不是很上头呀.预处理导包常规操作
- 首先感谢比尔、感谢微软、感谢MSDN,是他们让我看到他们富有创意的一面,好了好了不废话了。我们经常把多个CSS或者多个JS并成一个,以节省请
- 本教程旨在介绍如何使用七牛的Python SDK来快速地进行文件上传,下载,处理,管理等工作。安装首先,要使用Python的SDK必须要先安
- 1.以前的方法如果是要获得程序运行的当前目录所在位置,那么可以使用os模块的os.getcwd()函数。如果是要获得当前执行的脚本的所在目录
- 正则表达式在 PHP 中的应用在 PHP 应用中,正则表达式主要用于:•正则匹配:根据正则表达式匹配相应的内容•正则替换:根据正则表达式匹配
- PHP屏蔽蜘蛛访问代码代码:常用搜索引擎名与 HTTP_USER_AGENT对应值百度baiduspider谷歌googlebot搜狗sog