研究方法,python+虚拟机+https监视工具
#这只是一个半成品,只是实现了,一个完整的订票过程,对于,如果刷票,自己研究
#简单过程
# 第一、getRandAndCookie() 获得cook 和一个随机数用于登录
# 第二、getEnterRandCode() 得到登录时的识别码
# 第三、setuseandpassword(randcode,use,password) 发送随机数、识别码和用户及密码。由于随机数只在内部使用,所以定义成了全局变量,
# 第四、GetTrainList() 得到所有车站列表,'@bjb|北京北|VAP|beijingbei|bjb|0' 其中有中文、拼音、拼音缩写、所一个ID(唯一),其主要是可以,通过上面的列表,找到它的唯一ID,TranCityToId('南昌')
# 第五、GetTrainNumList(date,fromstationid,tostationid,starttime) 得到哪到哪的所在车次,消息格式如下,其中所以,一下车次的的ID:"id":"650000K1060I"
# {"end_station_name":"北京西","end_time":"16:18","id":"650000K1060I","start_station_name":"深圳","start_time":"10:54","value":"K106"}
# 通过ChangeToTrainNumId('K106')得到车次ID
# 第六、QueryTrain(fromstationid,tostationid,date,stationNum,starttime) 就是点击查询按键,得到是否有能预订,格式如下
# <span id='id_5700000Z6809'
# class='base_txtdiv'
# οnmοuseοver=javascript:onStopHover('5700000Z6809#NCG#BXP')
# οnmοuseοut='onStopOut()'>Z68</span>,<img src='/otsweb/images/tips/first.gif'> 南昌 <br> 20:12,<img src='/otsweb/images/tips/last.gif'> 北京西 <br> 07:38,11:26,--,--,--,--,10,<font color='#008800'>有</font>,<font color='#008800'>有</font>,--,<font color='#008800'>有</font>,<font color='#008800'>有</font>,--,<a name='btn130_2' class='btn130_2' style='text-decoration:none;'
# οnclick=javascript:getSelected('Z68#11:26#20:12#5700000Z6809#NCG#BXP#07:38#南昌#北京西#01#04#1*****31804*****00481*****03306*****00103*****0162#4AFCA0360B0B029AB6E00E3DF2231EE95B4CA798E4D91F296D9BF4CF#G1')>预 订</a>
# 通过choiceSubmitNum(stationNum,trainsubmitinfo)提取出getSelected()消息
# 第七、submitRequest(choiceSubmitNum(stationNum,trainsubmitinfo),date,starttime) 就是点击预订按钮
# 第八、getrandCheckCode()得到提交订单的识别码
# 第十、CheckInMyTicket(info,randcode,peoples)点击提交,如果成功的话,就会返回{"errMsg":"Y"}
# 出于,网络是UTF8格式,所以,必须# -*- coding: utf-8 -*-,(当然,自己转换也是可以的)
# 出于这一个控制台信息,所以,识别码的图片在脚本同一目录
把代码贴上。
# -*- coding: utf-8 -*-
import pycurl
import StringIO
import ConfigParser
#这只是一个半成品,只是实现了,一个完整的订票过程,对于,如果刷票,自己研究
#简单过程
# 第一、getRandAndCookie() 获得cook 和一个随机数用于登录
# 第二、getEnterRandCode() 得到登录时的识别码
# 第三、setuseandpassword(randcode,use,password) 发送随机数、识别码和用户及密码。由于随机数只在内部使用,所以定义成了全局变量,
# 第四、GetTrainList() 得到所有车站列表,'@bjb|北京北|VAP|beijingbei|bjb|0' 其中有中文、拼音、拼音缩写、所一个ID(唯一),其主要是可以,通过上面的列表,找到它的唯一ID,TranCityToId('南昌')
# 第五、GetTrainNumList(date,fromstationid,tostationid,starttime) 得到哪到哪的所在车次,消息格式如下,其中所以,一下车次的的ID:"id":"650000K1060I"
# {"end_station_name":"北京西","end_time":"16:18","id":"650000K1060I","start_station_name":"深圳","start_time":"10:54","value":"K106"}
# 通过ChangeToTrainNumId('K106')得到车次ID
# 第六、QueryTrain(fromstationid,tostationid,date,stationNum,starttime) 就是点击查询按键,得到是否有能预订,格式如下
# <span id='id_5700000Z6809'
# class='base_txtdiv'
# οnmοuseοver=javascript:onStopHover('5700000Z6809#NCG#BXP')
# οnmοuseοut='onStopOut()'>Z68</span>,<img src='/otsweb/images/tips/first.gif'> 南昌 <br> 20:12,<img src='/otsweb/images/tips/last.gif'> 北京西 <br> 07:38,11:26,--,--,--,--,10,<font color='#008800'>有</font>,<font color='#008800'>有</font>,--,<font color='#008800'>有</font>,<font color='#008800'>有</font>,--,<a name='btn130_2' class='btn130_2' style='text-decoration:none;'
# οnclick=javascript:getSelected('Z68#11:26#20:12#5700000Z6809#NCG#BXP#07:38#南昌#北京西#01#04#1*****31804*****00481*****03306*****00103*****0162#4AFCA0360B0B029AB6E00E3DF2231EE95B4CA798E4D91F296D9BF4CF#G1')>预 订</a>
# 通过choiceSubmitNum(stationNum,trainsubmitinfo)提取出getSelected()消息
# 第七、submitRequest(choiceSubmitNum(stationNum,trainsubmitinfo),date,starttime) 就是点击预订按钮
# 第八、getrandCheckCode()得到提交订单的识别码
# 第十、CheckInMyTicket(info,randcode,peoples)点击提交,如果成功的话,就会返回{"errMsg":"Y"}
# 出于,网络是UTF8格式,所以,必须# -*- coding: utf-8 -*-,(当然,自己转换也是可以的)
# 出于这一个控制台信息,所以,识别码的图片在脚本同一目录
#得到头信息
# -*- coding: utf-8 -*-
import pycurl
import StringIO
import ConfigParser
import Image
import os
import re
import time
import json
import sys
import math
import binascii
#这只是一个半成品,只是实现了,一个完整的订票过程,对于,如果刷票,自己研究
#简单过程
# 第一、getRandAndCookie() 获得cook 和一个随机数用于登录
# 第二、getEnterRandCode() 得到登录时的识别码
# 第三、setuseandpassword(randcode,use,password) 发送随机数、识别码和用户及密码。由于随机数只在内部使用,所以定义成了全局变量,
# 第四、GetTrainList() 得到所有车站列表,'@bjb|北京北|VAP|beijingbei|bjb|0' 其中有中文、拼音、拼音缩写、所一个ID(唯一),其主要是可以,通过上面的列表,找到它的唯一ID,TranCityToId('南昌')
# 第五、GetTrainNumList(date,fromstationid,tostationid,starttime) 得到哪到哪的所在车次,消息格式如下,其中所以,一下车次的的ID:"id":"650000K1060I"
# {"end_station_name":"北京西","end_time":"16:18","id":"650000K1060I","start_station_name":"深圳","start_time":"10:54","value":"K106"}
# 通过ChangeToTrainNumId('K106')得到车次ID
# 第六、QueryTrain(fromstationid,tostationid,date,stationNum,starttime) 就是点击查询按键,得到是否有能预订,格式如下
# <span id='id_5700000Z6809'
# class='base_txtdiv'
# οnmοuseοver=javascript:onStopHover('5700000Z6809#NCG#BXP')
# οnmοuseοut='onStopOut()'>Z68</span>,<img src='/otsweb/images/tips/first.gif'> 南昌 <br> 20:12,<img src='/otsweb/images/tips/last.gif'> 北京西 <br> 07:38,11:26,--,--,--,--,10,<font color='#008800'>有</font>,<font color='#008800'>有</font>,--,<font color='#008800'>有</font>,<font color='#008800'>有</font>,--,<a name='btn130_2' class='btn130_2' style='text-decoration:none;'
# οnclick=javascript:getSelected('Z68#11:26#20:12#5700000Z6809#NCG#BXP#07:38#南昌#北京西#01#04#1*****31804*****00481*****03306*****00103*****0162#4AFCA0360B0B029AB6E00E3DF2231EE95B4CA798E4D91F296D9BF4CF#G1')>预 订</a>
# 通过choiceSubmitNum(stationNum,trainsubmitinfo)提取出getSelected()消息
# 第七、submitRequest(choiceSubmitNum(stationNum,trainsubmitinfo),date,starttime) 就是点击预订按钮
# 第八、getrandCheckCode()得到提交订单的识别码
# 第十、CheckInMyTicket(info,randcode,peoples)点击提交,如果成功的话,就会返回{"errMsg":"Y"}
# 出于,网络是UTF8格式,所以,必须# -*- coding: utf-8 -*-,(当然,自己转换也是可以的)
# 出于这一个控制台信息,所以,识别码的图片在脚本同一目录
#得到头信息
def headerCookie(buf):
global Cookie
# print buf
findindex = buf.find("Set-Cookie:")
if findindex >= 0:
beginindex = buf.find(':')
endindex = buf.find(';')
if endindex >0:
Cookie+=buf[beginindex+1:endindex+1]
#用于测试
def gethead(buf):
#
None
#print buf
def setcer(curl):
curl.setopt(pycurl.CAINFO, r"srca.cer") #根证书
None
##得到随机数和cookie连接参数
def getRandAndCookie():
global loginRand
global Cookie
Cookie = ''
loginRand = ''
curl = pycurl.Curl()
b = StringIO.StringIO()
curl.setopt(pycurl.URL, "https://www.12306.cn/otsweb/loginAction.do?method=loginAysnSuggest")
curl.setopt(pycurl.WRITEFUNCTION, b.write)
curl.setopt(pycurl.SSL_VERIFYPEER, 0)
curl.setopt(pycurl.SSL_VERIFYHOST, 0)
curl.setopt(pycurl.HEADERFUNCTION, headerCookie)
#curl.setopt(pycurl.COOKIE,Cookie)
curl.perform()
if curl.getinfo(pycurl.RESPONSE_CODE) == 200:
back = b.getvalue()
login = back.find('loginRand')
if login >=0:
begin =login + back[login:].find(':')
begin += back[begin:].find('"')
end = back[begin+1:].find('"')
loginRand = back[begin+1:end+begin+1]
curl.close()
return loginRand
def getEnterRandCode():
#得到登录验证码图
curl = pycurl.Curl()
f = StringIO.StringIO()
curl.setopt(pycurl.URL, "https://dynamic.12306.cn/otsweb/passCodeNewAction.do?module=login&rand=sjrand")
#更改了 以前是https://www.12306.cn/otsweb/passCodeAction.do?rand=sjrand")
curl.setopt(pycurl.WRITEFUNCTION, f.write)
curl.setopt(pycurl.SSL_VERIFYPEER, 0)
curl.setopt(pycurl.SSL_VERIFYHOST, 0)
curl.setopt(pycurl.HEADERFUNCTION, headerCookie)
curl.setopt(pycurl.COOKIE,Cookie)
curl.perform()
backinfo = ''
if curl.getinfo(pycurl.RESPONSE_CODE) == 200:
backinfo = f.getvalue()
curl.close()
return backinfo
def GetKeyhttp():
curl = pycurl.Curl()
f = StringIO.StringIO()
curl.setopt(pycurl.URL, "https://dynamic.12306.cn/otsweb/loginAction.do?method=init")
curl.setopt(pycurl.WRITEFUNCTION, f.write)
curl.setopt(pycurl.SSL_VERIFYPEER, 0)
curl.setopt(pycurl.SSL_VERIFYHOST, 0)
curl.setopt(pycurl.HEADERFUNCTION, gethead)
curl.setopt(pycurl.COOKIE,Cookie)
curl.setopt(pycurl.ENCODING, 'gzip, deflate')
curl.setopt(pycurl.HTTPHEADER,["Accept-Language:zh-CN",
"Referer:https://dynamic.12306.cn/otsweb/",
"Accept:*/*",
"Accept-Encoding:gzip, deflate",
"Connection:Keep-Alive"])
curl.perform()
backinfo = ''
if curl.getinfo(pycurl.RESPONSE_CODE) == 200:
backinfo = f.getvalue()
curl.close()
begin = backinfo.find('/otsweb/dynamicJsAction.do?jsversion=')
end = backinfo[begin+1:].find('"') + begin+1
return "https://dynamic.12306.cn"+backinfo[begin:end]
#/otsweb/dynamicJsAction.do?jsversion=4296&method=loginJs
return backinfo
def GetNewKey():
http = GetKeyhttp()
print http
curl = pycurl.Curl()
f = StringIO.StringIO()
curl.setopt(pycurl.URL, http)
curl.setopt(pycurl.WRITEFUNCTION, f.write)
curl.setopt(pycurl.SSL_VERIFYPEER, 0)
curl.setopt(pycurl.SSL_VERIFYHOST, 0)
curl.setopt(pycurl.HEADERFUNCTION, gethead)
curl.setopt(pycurl.COOKIE,Cookie)
curl.setopt(pycurl.ENCODING, 'gzip, deflate')
curl.setopt(pycurl.HTTPHEADER,["Accept-Language:zh-CN",
"Referer:https://dynamic.12306.cn/otsweb/loginAction.do?method=init",
"Accept:*/*",
"Accept-Encoding:gzip, deflate",
"Connection:Keep-Alive"])
curl.perform()
backinfo = ''
if curl.getinfo(pycurl.RESPONSE_CODE) == 200:
backinfo = f.getvalue()
curl.close()
fkey = backinfo.find('function gc()')
if fkey > 0:
beginkey = backinfo[fkey:].find("'") + fkey
end = backinfo[beginkey+1:].find("'") + beginkey+1
return backinfo[beginkey+1:end].replace('=','%3D')+'='+(Get12306NewKey(backinfo[beginkey+1:end]).replace('=','%3D'))
def stringToLongArray(string, includeLength):
length = len(string)
result = []
i = 0;
while (i < length):
re = 0
try:
re = ord(string[i:i+1])
re = re | (ord(string[i+1:i+2] )<< 8)
re = re | (ord(string[i+2:i+3] )<< 16)
re = re | (ord(string[i+3:i+4] )<< 24)
except:
pass
result.append(re)
i += 4
if (includeLength):
result.append(length)
return result
def longArrayToString(data,includeLength):
length = len(data)
n = (length - 1) << 2
if (includeLength):
m = data[length - 1]
if((m < n - 3) or (m > n)):
return None
n = m
bdata = ""
for i in range(0,length):#(var i = 0;i < length;i ++ ):
bdata = bdata + chr(data[i] & 0xff) + chr(data[i] >> 8 & 0xff) + chr(data[i] >> 16 & 0xff) + chr(data[i] >> 24 & 0xff)
if (includeLength):
return bdata[0:n]
else:
return bdata
def Base32encrypt(string, key):
if (string == ""):
return "";
delta = 0x9E3779B8
v = stringToLongArray(string, True)
k = stringToLongArray(key, False)
k_length = len(k)
if (k_length < 4):
k.append(0)
k.append(0)
k.append(0)
k_length = 4
v_length = len(v)
n = v_l