本文章中所有内容仅供学习交流,抓包内容、敏感网址、数据接口均已做脱敏处理,严禁用于商业用途和非法用途,否则由此产生的一切后果均与作者无关,若有侵权,请联系我立即删除!
目标
解决JD常用的验证码
前言
这里常用的就三种接口
[ol]
[/ol]
还有两个验证码
三种验证
登录的滑块接口
接口链接
aHR0cHM6Ly9wYXNzcG9ydC5qZC5jb20vdWMvbG9naW4/bHR5cGU9bG9nb3V0
data:image/s3,"s3://crabby-images/717eb/717eb586a6308b3a5d5e5a32e677385faa2639f2" alt=""
如上图所示。
这里就不多说废话了。直接刷新看接口
data:image/s3,"s3://crabby-images/6d0b3/6d0b38bbff33d0a7fc5df19ac900cb90306f1b56" alt=""
这里可以看到 有两个接口请求发出。分别为
g请求
这里看g请求。直接进第二个栈。
data:image/s3,"s3://crabby-images/215a1/215a1cf824ae55e6f2b0d01b08a68c759237bf62" alt=""
这里看到 很多值已经给了我们值了。这里我们重点观察两个值。分别是e 和 j
data:image/s3,"s3://crabby-images/0f633/0f63321c36067e19971b427ebe3dd8610b2d16e5" alt=""
这里打断点进栈
data:image/s3,"s3://crabby-images/2a3ca/2a3cac6e96de67e54bf7dc35c91e14d113b52d45" alt=""
好了 然后再进去。然后发现到一个虚拟机里了 这里包含了 e的生成。
data:image/s3,"s3://crabby-images/f37e2/f37e29bda4d9b957fe51e7567e275bcb37f24f14" alt=""
然后就是j
data:image/s3,"s3://crabby-images/74b91/74b91f9691ae18959a57b5bfd62413af6c91d9f3" alt=""
j的生成不必多说。看下图就知道了。
data:image/s3,"s3://crabby-images/82f63/82f630f3ee9206b710ff89d61ff8630c35ef636d" alt=""
其他的值。
然后看返回值
data:image/s3,"s3://crabby-images/bebf8/bebf805cdca117f8e56e6cc73981fe569bf62a48" alt=""
上述请求有几个值需要提出来。方便后续s调用
S请求
这里继续看s请求。
data:image/s3,"s3://crabby-images/69579/69579a6245281cfe18ae2da74a2d6aa3e51f9a03" alt=""
这里和上图结果图作为比较。可以发现。
c好像就是challenge。
这里话不多说了。其他的值都不重要 。
最主要是这个d参数。
我们重点来看这个d
data:image/s3,"s3://crabby-images/f4d1d/f4d1d855db16beac35f26fdf92272eb3a040e15a" alt=""
如下图所示。很多值的生成都能直接看到。
这里d 值的生成 则是由一个方法传参加密了一堆数组。
data:image/s3,"s3://crabby-images/2c91e/2c91e62f5cac22d35032a55ff075f3a3fc528c80" alt=""
那这个数组是啥东西。其实我们不难发现这就是个轨迹。通过getCoordinate这个方法 才变成了请求接口中的d这个样子。
d方法如下:
'getCoordinate': function(a) {
var b = this;
var c = new Array();
for (var d = 0x0; d
这个方法很简单。缺什么扣什么。
然后传值轨迹。
至于轨迹的生成。可以通过某些算法去实现这个列表数组的生成。
这里是通过举例然后偏移出数组变量
这里贴个方法用于生成数组。
def offer(distance):
index = 0
slide = []
indexTime = str(int(time.time()))[:9]
for item in self.base_slide:
index += 1
item[2] = int(indexTime + str(item[2])[-4:])
if int(item[0]) >= (distance + int(self.base_slide[0][0])):
slide = self.base_slide[:index]
slide.append(
[str(distance + int(self.base_slide[0][0])), item[1], item[2] + 700 + int(random.random() * 1000)])
break
last = int(slide[-1][0].split('.')[0])
pIndex = 0
for item in self.push_slide:
if pIndex == 0 or pIndex == len(self.push_slide) - 1:
times = slide[-1][2]
else:
times = slide[-1][2] + (self.push_slide[pIndex + 1][2] - self.push_slide[pIndex][2])
slide.append([str(item[0] + last), '369', times])
pIndex += 1
# print(json.dumps(slide))
return slide
最后通过getCoordinate 生成加密值。
最后伪装成功参数 然后请求就行了 返回下图代表成功。
data:image/s3,"s3://crabby-images/b364f/b364f41ec39e3c0036d0ec25ec5fe20d1b655a41" alt=""
cfe接口滑块
下面接口网站如下
aHR0cHM6Ly9jZmUubS5qZC5jb20vcHJpdmF0ZWRvbWFpbi9yaXNrX2hhbmRsZXIvMDMxMDE5MDAvP3JldHVybnVybD1odHRwcyUzQSUyRiUyRml0ZW0uamQuY29tJTJGMTAwMDQ4MjcyNzYyLmh0bWwmcnFob3N0PWh0dHBzJTNBJTJGJTJGYXBpLm0uamQuY29tJnJwaWQ9cnAtMTg2NTQ5NDkwLTEwMDU2LTE3MDk3MTMyMDgzNzEmZXZ0eXBlPTImZXZhcGk9Y29sb3JfcGNfZGV0YWlscGFnZV93YXJlQnVzaW5lc3Mmc291cmNlPTEmZm9yY2VDdXJyZW50Vmlldz0x
如下图页面。这就到了另一个滑块的接口。
data:image/s3,"s3://crabby-images/d8c47/d8c47228245919a0253fcff61ecb103a481ee173" alt=""
这里我们点击快速验证。然后完整的走一遍流程看看。具体的走向是如何的
如下图所示。
data:image/s3,"s3://crabby-images/bd321/bd321ee099db755380d631b2f3ef26b485d5ece6" alt=""
这里首先筛选xhr请求。请求太多
如下图所示。图中经过了这么多的请求。
data:image/s3,"s3://crabby-images/d5009/d50095483658f684486835efb43a3d44f7b4a995" alt=""
我们一个一个看。
第一个m?std 请求 返回值是0 目测只是判断状态的请求。
第二个api请求。如下图 返回了一个值。请求参数中enbody加密了
data:image/s3,"s3://crabby-images/1dd73/1dd73592e73b73b3c3d1d531b7d29f0310af1cf8" alt=""
data:image/s3,"s3://crabby-images/03066/030666e8f1161fbd38c8c0dc81f3b92250f4a9e8" alt=""
第三个api请求也是加密了 enbody。返回值没东西。应该也是个请求判定。
第四个fp请求。请求需要伪装si 和 ct。 而这个si 刚好和第二个api请求的si相同。 返回了 fp 和 st。
data:image/s3,"s3://crabby-images/b4b5a/b4b5a2a692d0406342b281cdcee6bddbe4ecc0c2" alt=""
data:image/s3,"s3://crabby-images/23fda/23fdaadb6bc67dc415b754143e9ac83b34021b5b" alt=""
第五个请求: web_jcap_report 返回了 code为0 无用请求。
第六个请求:请求传参中带有上图fp参数的加密 还多了个tk加密。返回值 返回了 缺口图和背景图。
data:image/s3,"s3://crabby-images/d2bac/d2baca54c9f5d5090555e73b1cbad8d569b9057d" alt=""
data:image/s3,"s3://crabby-images/05bc0/05bc0a2b27975788743ca08b3cf83c089d414bf4" alt=""
第七个请求:无用请求
第八个请求:即是第二个check请求。同第六个请求相似。但是返回值不一样。
那这样我们大概就能明白个具体流程了。
[ol]
[/ol]
api请求
这里我们首先去看一下api请求。
data:image/s3,"s3://crabby-images/3d98c/3d98cd7554b26f529eb3d34964e764b00d38e54e" alt=""
如上图。所示 只有这个enbody值是通过加密获得的。那我们去搜索一下 或者走栈。
data:image/s3,"s3://crabby-images/5079e/5079ed466488a7321a1504dc0430cff38a2b0a68" alt=""
如下图所示 直接获取到了 enbody加密的地方。
data:image/s3,"s3://crabby-images/9f058/9f058aad5dec456bc37f185815f53785009c1c73" alt=""
这里简化一下代码
result = encrypt(JSON.stringify(param), 'rhiasnkdhandrisk', 'r-s-h-n_r_isnkdk')
然后过一下断点。这里可以看到 加密的值分别为
eid: 可以为空
evType: 应该是验证码类型 2
requestId: 由另一个接口605返回的requestID
shshshfpx: 是一个随机值的算法。包含随机生成以及时间戳的拼接。
data:image/s3,"s3://crabby-images/a6d95/a6d95f6d9445ecd67a4707e10d25c5e434035384" alt=""
第二次api请求。这里第二次请求加密的值是 第一次的返回值 已经第一次的那个请求参数。其实可以不需要。这里也展示一下
data:image/s3,"s3://crabby-images/61111/61111e13940452a1d8e9b5093a70cd5909e9aaf3" alt=""
fp请求
这里fp请求。不太好搜索。我们直接进栈
data:image/s3,"s3://crabby-images/ea56a/ea56a89f1af7a39d8d68ad2693e869ba5fd13757" alt=""
进栈然后搜索 .si 这里可以发现。ct的算法 已经出来了。
data:image/s3,"s3://crabby-images/42757/42757d9f627c7508d66f2fd7c0db0628c6b5abef" alt=""
data:image/s3,"s3://crabby-images/734cc/734ccb5d2b42f2712b5c31983da1a3a65998386a" alt=""
这里可以看到 。这个值应该就是加密的值。
然后把几个函数互相扣一下就行。
这个x函数的算法如下
data:image/s3,"s3://crabby-images/afe3f/afe3f5645dd687ec915fb6520e2ad91af542dab5" alt=""
function s(t, e, n) {
var r = t
, a = o;
e && (i = e);
return c(r, a)
}
这里层层扣即可。
然后把第一次api返回的那个data和这个新生成的ct
去请求 得到fp和 st
两次check
第一次
断点继续走 走到下图所在位置
data:image/s3,"s3://crabby-images/8bd8a/8bd8aebedc67af7cd6cbdba8f2350975514676bb" alt=""
如上图所示。
这里需要注意的是 这里的t 暂时是不传值的。这里t是什么先留个悬念。
data:image/s3,"s3://crabby-images/b385c/b385cbf49d9b1df2ee6179a0c1fcbc9009ca5856" alt=""
然后进行加密就得到了tk和ct。
然后请求。得到两张以base64为编码的图片。
第二次check请求
这里其他值加密都是一样的。
唯一不同就是这个 JSON.stringify(t) 如下图所示
data:image/s3,"s3://crabby-images/aec0b/aec0bc569274e6f442469c78d47285b24c67cb0a" alt=""
这里传值是轨迹。通过两张图片识别出来的距离 最终通过算法生成轨迹。
这里轨迹简单看看 第一个是移动的距离值。
data:image/s3,"s3://crabby-images/9afaa/9afaa3017030c56f033f67bc157e9bd9123fe6ac" alt=""
第二个和第三个 如下图js所示的位置
data:image/s3,"s3://crabby-images/490f5/490f53f056f305c16c987997ab1d92e3fb1cdc6b" alt=""
这里可以通过算法去实现 也可以通过自建库来实现。
最后通过这些值加密 得到请求参数即可完成滑块。整体流程如下
data:image/s3,"s3://crabby-images/5d101/5d1010154662f7edf634927b422e392e9be7c90a" alt=""
cfe 点选验证
下面接口网站如下
aHR0cHM6Ly9jZmUubS5qZC5jb20vcHJpdmF0ZWRvbWFpbi9yaXNrX2hhbmRsZXIvMDMxMDE5MDAvP3JldHVybnVybD1odHRwcyUzQSUyRiUyRml0ZW0uamQuY29tJTJGMTAwMDQ4MjcyNzYyLmh0bWwmcnFob3N0PWh0dHBzJTNBJTJGJTJGYXBpLm0uamQuY29tJnJwaWQ9cnAtMTg2NTQ5NDkwLTEwMDU2LTE3MDk3MTMyMDgzNzEmZXZ0eXBlPTImZXZhcGk9Y29sb3JfcGNfZGV0YWlscGFnZV93YXJlQnVzaW5lc3Mmc291cmNlPTEmZm9yY2VDdXJyZW50Vmlldz0x
和滑块一个连接 不同的是 验证码不一样。
触发方式应该是与爬取的等级而定。
data:image/s3,"s3://crabby-images/558eb/558eb8b1d7a42b8479d0ed0ada7fb5fa99b07145" alt=""
如上图所示。
其实上图大体流程都是一样的。这里唯一有一点区别。
如下图所示。这里JSON.stringify(t) 里面传值的不是轨迹。而是 点选的坐标。
data:image/s3,"s3://crabby-images/8f505/8f50507f9db6489c66ebf0835146fe3712e7b604" alt=""
这里可以通过两种方式去解决
[ol]
[/ol]
这里还有一点需要注意。
这里点选的 验证码
第一次check返回的tp值是22 。
而滑块是30
data:image/s3,"s3://crabby-images/79c97/79c97ac4877aa285efae32d859a849853c037714" alt=""
可以通过这个去区分验证码。