事情是这么开始的。国庆回家和亲戚欢聚一堂,开心嘛,天然要打麻将。php
打了几圈下来,一不当心赢了八大姑10块钱。躺在床上,正在美滋滋地数钱。html
忽然八大姑一个微信找过来,说他的同事的媳妇的妹妹,要参加一个XXX比赛。要刷票!前端
亲情,伦理,普世价值观一瞬间在心头涌现,手里的钱它忽然就不香了。git
在这一时刻,我作出了一名党员应有的判断。github
拿人钱财,替人消灾。golang
就像江湖,只要有投票存在的地方,有会有刷票的人。ajax
《马克思主义基本原理》中应该对这个现象应该有过详细的解释,只是马原的课上我大几率在睡觉,也没怎么听。反正人类社会就没有马原不能解释的🐶。docker
人们习惯把刷票的人,称为刷子。就好比说我。有时候遇到对手,比拼刷票速度,排行榜上会出现两个票数奇高的选手,你方唱罢我登场,疯狂涨票。json
人们惊呼,有两把刷子。浏览器
直接在PC端打开页面,发现有一层微信浏览器的验证。
种状况,一般多是ua的验证,或者是须要受权openId的验证。先绕过UA试试。把UA修改成:
Mozilla/5.0 (Linux; U; Android 2.3.6; zh-cn; GT-S5660 Build/GINGERBREAD) AppleWebKit/533.1 (KHTML, like Gecko) Version/4.0 Mobile Safari/533.1 MicroMessenger/4.5.255
一发入魂,进入投票界面。投一票试试:
POST: https://op.bookan.com.cn/index.php
-------------------------------
FormData:
op: ReadScreen.operateSoundLimit
instanceId: 0
weChatId:
id: 1377928
level: 2
uniCode: fbHCGPyaraaRg64oo4tt1570604768281
type: 2
key: SXPg6VhOYYeCsiTrDvZZensypfwhbi
token: 3b18fb13165cc5c86683fcb5ce221b29
复制代码
其它参数都很好理解,uniCode / key / token 这三个参数在这种时候就显得很是灵性了。显然,这个页面是作了加密参数的校验。
全文搜索 operateSoundLimit,找到这个ajax的请求源,打上断点,点击。找到了这个请求的发出地址。
不断反复debug,找到这几个参数的生成函数。因为js是通过压缩混淆,看不懂,不要紧,粘出来。
TokenJs := `function t(n,t){var r=(65535&n)+(65535&t);return(n>>16)+(t>>16)+(r>>16)<<16|65535&r}function r(n,t){return n<<t|n>>>32-t}function e(n,e,o,u,c,f){return t(r(t(t(e,n),t(u,f)),c),o)}function o(n,t,r,o,u,c,f){return e(t&r|~t&o,n,t,u,c,f)}function u(n,t,r,o,u,c,f){return e(t&o|r&~o,n,t,u,c,f)}function c(n,t,r,o,u,c,f){return e(t^r^o,n,t,u,c,f)}function f(n,t,r,o,u,c,f){return e(r^(t|~o),n,t,u,c,f)}function i(n,r){n[r>>5]|=128<<r%32,n[14+(r+64>>>9<<4)]=r;var e,i,a,d,h,l=1732584193,g=-271733879,v=-1732584194,m=271733878;for(e=0;e<n.length;e+=16)i=l,a=g,d=v,h=m,g=f(g=f(g=f(g=f(g=c(g=c(g=c(g=c(g=u(g=u(g=u(g=u(g=o(g=o(g=o(g=o(g,v=o(v,m=o(m,l=o(l,g,v,m,n[e],7,-680876936),g,v,n[e+1],12,-389564586),l,g,n[e+2],17,606105819),m,l,n[e+3],22,-1044525330),v=o(v,m=o(m,l=o(l,g,v,m,n[e+4],7,-176418897),g,v,n[e+5],12,1200080426),l,g,n[e+6]...`
TokenExecJs := `var zzzaaa = A('BO0KanIsGoOd%s')`
复制代码
随便找个js执行引擎:otto
func GetToken(key string) string {
vm := otto.New()
_, err := vm.Run(TokenJs)
if err != nil {
panic(err)
}
vm.Run(fmt.Sprintf(TokenExecJs, key))
value, _ := vm.Get("zzzaaa")
return value.String()
}
复制代码
用PostMan导入脚本测试了几下以后,发现返回竟然变了,里面带上了一个验证码图片的Base64。须要二次请求验证。
至于这个验证码,太简单了,直接接个OCR,就搞定: gosseract
// 抽取返回中的 key 和 验证码base64
func GetCaptureUrl(str string) (key string, pngBase64 string) {
defer func() {
if err := recover(); err != nil {
}
}()
bytes := []byte(str)
data := map[string]interface{}{}
_ = json.Unmarshal(bytes, &data)
key = strconv.FormatFloat(data["data"].(map[string]interface{})["key"].(float64), 'f', -1, 64)
pngBase64 = strings.TrimSpace(strings.ReplaceAll(data["data"].(map[string]interface{})["image"].(string), "data:image/png;base64,", ""))
return
}
// 识别结果
func ReadPng(data string) (id, valus string) {
key, pngBase64 := "", ""
for key == "" {
key, pngBase64 = GetCaptureUrl(data)
}
b, _ := base64.StdEncoding.DecodeString(pngBase64)
client := gosseract.NewClient()
defer client.Close()
client.SetImageFromBytes(b)
text, _ := client.Text()
re, _ := regexp.Compile("[^0-9]") // 因为验证码只有数字,去掉非数字字符
text = re.ReplaceAllString(text, "")
return key, text
}
复制代码
刷了一段时间以后,发现单IP到达100票左右的时候,继续投也没法成功了。
上代理IP池! github.com/awolfly9/IP… 最怕的就是搭环境,直接找个热心人士编译的docker run起来。 registry.hub.docker.com/u/yeclimeri…
写一段获取代理IP的方法:
func GetProxy() []string {
var s []string
resp, _ := http.Get(ProxyPoolUrl)
reader, _ := simplejson.NewFromReader(resp.Body)
array, _ := reader.Array()
for _, a := range array {
s = append(s, a.(string))
}
fmt.Println("proxy:", s)
return s
}
复制代码
刷票的车轮就飞快地运转起来了~
刷票一时爽啊 T.T。逃...
有一些投票特别严格,须要人脸验证。或者是须要银行卡实名认证。基本就没办法经过程序自动化完成了。这种严格的投票,票数一般也不会特别多。
这时候,你能够充分享受中国的廉价劳动力以及人口红利带来的优点。手动狗头。
祭出法宝:猪八戒大法。
欢迎风控,反做弊的同窗提问~
代码: code.byted.org/niejunhao/t…
收费代理池: 收费代理池推荐
短信验证码: 接码平台推荐