#!/usr/bin/env python

# -*- encoding:utf-8 -*-

'''

Created on 2014年12月18日

@author: liujichao

'''

import datetime

import json

import re

import sys

import time

import Image

import PyV8

import requests

import tools.email_helper as emailHelper

reload(sys)

sys.setdefaultencoding('utf-8')# @UndefinedVariable

reqSingle= requests.Session()

attCheCi=["G655","G6741","G67","G491"]#关注的车次

dateList=["2015-02-18"]#关注的日期

username="12306登录用户名"

password="登录密码"

#这个是需要手动提交订单后f12自己找的,挨个post请求去找,参数名为:oldPassengerStr 格式如下

oldPassengerStr="姓名,1,130434199802036011,1_姓名2,1,130434199204238069,1_"

#这个是需要手动提交订单后f12自己找的,挨个post请求去找,参数名为:passengerTicketStr 格式如下

passengerTicketStr="O,0,1,姓名,1,130434199802036011,13683456789,N_O,0,1,姓名2,1,130434199204238069,13683456789,N"

header={

"Accept":"text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8",

"Accept-Encoding":"gzip, deflate",

"Accept-Language":"zh-cn,zh;q=0.8,en-us;q=0.5,en;q=0.3",

"Connection":"keep-alive",

"Host":"kyfw.12306.cn",

"Referer":"https://kyfw.12306.cn/otn/safeguard/init",

"User-Agent":"Mozilla/5.0 (Windows NT 5.1; rv:34.0) Gecko/20100101 Firefox/34.0"

}

##定火车票

def orderTicket(fromStation,toStation,trainDate,secretStr):

header["Referer"]="https://kyfw.12306.cn/otn/leftTicket/init"

orderInitReq= reqSingle.get("https://kyfw.12306.cn/otn/leftTicket/init",headers=header)

header["Referer"]="https://kyfw.12306.cn/otn/leftTicket/init"

aryKV=extractKey(orderInitReq.content,header)

print aryKV

#初始化订票

header["Referer"]="https://kyfw.12306.cn/otn/leftTicket/init"

orderInitReq= reqSingle.post("https://kyfw.12306.cn/otn/leftTicket/submitOrderRequest",data={

aryKV[0]:aryKV[1],

"train_date":trainDate,

"myversion":"undefined",

"purpose_codes":"ADULT",

"query_from_station_name":fromStation,

"query_to_station_name":toStation,

"secretStr":secretStr,

"tour_flag":"dc",

"back_train_date":time.strftime('%Y-%m-%d',time.localtime(time.time())),

"undefined":""

},headers=header)

print orderInitReq.content

orderInitJson=orderInitReq.json()

if orderInitJson.get("status")==False or  orderInitJson.get("httpstatus")!=200:

raise Exception("订票出现错误")

initDcReq= reqSingle.post("https://kyfw.12306.cn/otn/confirmPassenger/initDc", data={"_json_att":""},headers=header)

header["Referer"]="https://kyfw.12306.cn/otn/confirmPassenger/initDc"

aryKV=extractKey(initDcReq.content,header)

match=re.search("var globalRepeatSubmitToken = '(.*?)';", initDcReq.content)

ticketToken=match.group(1)

lianxirenReq=reqSingle.post("https://kyfw.12306.cn/otn/confirmPassenger/getPassengerDTOs", data={"REPEAT_SUBMIT_TOKEN":ticketToken,"_json_att":""},headers=header)

lianxirenJson=lianxirenReq.json()

#验证码

#开始做验证码

while True:

r=reqSingle.get("https://kyfw.12306.cn/otn/passcodeNew/getPassCodeNew?module=passenger&rand=randp&",verify=False,timeout=5,headers=header)

withopen("orderRand.jpg","wb") as rimg:

rimg.write(r.content)

pass

img=Image.open("orderRand.jpg")

img.show()

randCode=raw_input("请输入登录验证码:")

#验证验证码

randReq= reqSingle.post("https://kyfw.12306.cn/otn/passcodeNew/checkRandCodeAnsyn",data={

"REPEAT_SUBMIT_TOKEN":ticketToken,

"_json_att":"",

"rand":"randp",

"randCode":randCode},headers=header)

randRes=randReq.json()

if randRes.get("status")and randRes.get("httpstatus")==200 and randRes.get("data").get("result")=="1":

break;

pass

print "验证码输入正确!"

#检查票

checkOrderInfoReq=reqSingle.post("https://kyfw.12306.cn/otn/confirmPassenger/checkOrderInfo", data={

aryKV[0]:aryKV[1],

"REPEAT_SUBMIT_TOKEN":ticketToken,

"_json_att":"",

"bed_level_order_num":"000000000000000000000000000000",

"cancel_flag":2,

"oldPassengerStr":oldPassengerStr,

"passengerTicketStr":passengerTicketStr,

"randCode":randCode,

"tour_flag":"dc"

})

checkOrderInfoJson=checkOrderInfoReq.json()

if checkOrderInfoJson.get("status")==False or checkOrderInfoJson.get("httpstatus")!=200:

raise Exception("检查票出现错误")

pass

fromStationTelecode=re.search("'from_station_telecode':'(.*?)'", initDcReq.content).group(1)

leftTicket=re.search("'ypInfoDetail':'(.*?)'", initDcReq.content).group(1)

purpose_codes=re.search("'purpose_codes':'(.*?)'", initDcReq.content).group(1)

station_train_code=re.search("'station_train_code':'(.*?)'", initDcReq.content).group(1)

to_station_telecode=re.search("'to_station_telecode':'(.*?)'", initDcReq.content).group(1)

train_no=re.search("'train_no':'(.*?)'", initDcReq.content).group(1)

queueCountReq=reqSingle.post("https://kyfw.12306.cn/otn/confirmPassenger/getQueueCount",data={

"REPEAT_SUBMIT_TOKEN":ticketToken,

"_json_att":"",

"fromStationTelecode":fromStationTelecode,

"leftTicket":leftTicket,

"purpose_codes":purpose_codes,

"seatType":0,

"stationTrainCode":station_train_code,

"toStationTelecode":to_station_telecode,

"train_date":datetime.datetime.fromtimestamp(time.mktime(time.strptime(trainDate,'%Y-%m-%d'))).strftime('%a %b %d %Y %H:%M:%S GMT+0800'),

"train_no":train_no

},headers=header)

queueCountJson=queueCountReq.json()

print queueCountReq.content

if queueCountJson.get("status")==False or queueCountJson.get("httpstatus")!=200:

raise Exception("获取队列错误")

#确认队列

key_check_isChange=re.search("'key_check_isChange':'(.*?)'", initDcReq.content).group(1)

train_location=re.search("'train_location':'(.*?)'", initDcReq.content).group(1)

singleForQueueReq=reqSingle.post("https://kyfw.12306.cn/otn/confirmPassenger/confirmSingleForQueue",data={

"REPEAT_SUBMIT_TOKEN":ticketToken,

"_json_att":"",

"dwAll":"N",

"key_check_isChange":key_check_isChange,

"leftTicketStr":leftTicket,

"oldPassengerStr":oldPassengerStr,

"passengerTicketStr":passengerTicketStr,

"purpose_codes":purpose_codes,

"randCode":randCode,

"train_location":train_location

},headers=header)

singleForQueueJson=singleForQueueReq.json()

print singleForQueueReq.content

if singleForQueueJson.get("status")==False or singleForQueueJson.get("httpstatus")!=200:

raise Exception("confirmSingleForQueue异常")

if singleForQueueJson.get("data")is None or singleForQueueJson.get("data").get("submitStatus")==False:

raise Exception("confirmSingleForQueue异常")

#等待orderid

while True:

orderWaitReq= reqSingle.get("https://kyfw.12306.cn/otn/confirmPassenger/queryOrderWaitTime",data={"REPEAT_SUBMIT_TOKEN":ticketToken,

"_json_att":"",

"random":time.time(),

"tourFlag":"dc"

},headers=header)

print orderWaitReq.content

orderWaitJson=orderWaitReq.json()

if orderWaitJson.get("status")and orderWaitJson.get("httpstatus")==200:

if orderWaitJson.get("data")is not None and orderWaitJson.get("data").get("orderId")is not None:

orderId=orderWaitJson.get("data").get("orderId")

break

pass

pass

#进入队列

dcQueueReq=reqSingle.post("https://kyfw.12306.cn/otn/confirmPassenger/resultOrderForDcQueue",data={

"REPEAT_SUBMIT_TOKEN":ticketToken,

"_json_att":"",

"orderSequence_no":orderId

}

,headers=header)

dcQueueJson=dcQueueReq.json()

if dcQueueJson.get("status")and dcQueueJson.get("httpstatus")==200 and dcQueueJson.get("data")is not None and dcQueueJson.get("data").get("submitStatus"):

print "订票成功"

pass

else:

print dcQueueJson.content

print "订票失败"

pass

#https://kyfw.12306.cn/otn/confirmPassenger/resultOrderForDcQueue

pass

#检查是否登录

def checkIsLogin():

checkReq= reqSingle.post("https://kyfw.12306.cn/otn/login/checkUser", data={"_json_att":""},headers=header)

print u"检查是否登录"+checkReq.content

checkReqJson=checkReq.json()

if checkReqJson.get("status")and checkReqJson.get("httpstatus")==200:

if checkReqJson.get("data")is not None and checkReqJson.get("data").get("flag"):

return True

pass

return False

pass

#提取js加密内容后的key和value

def extractKey(htmlContent,headerxx):

loginMatch=re.search(r'', htmlContent)

jsUrl="https://kyfw.12306.cn"+ loginMatch.group(1)

jsReq=reqSingle.get(jsUrl,verify=False,timeout=15,headers=headerxx)

jsContent= jsReq.content

jsMatch=re.search("(function bin216.*?)function aj", jsContent)

jsEncode= jsMatch.group(1)#获取加密的js内容

keyMatch=re.search("var key='(.*?)'",jsContent)

loginKey= keyMatch.group(1)#获取登录的key

ctx=PyV8.JSContext()

ctx.enter()

ctx.eval(jsEncode)

loginValue=ctx.locals.encode32(ctx.locals.bin216(ctx.locals.Base32.encrypt("1111",loginKey)))

return [loginKey,loginValue]

pass

#登录操作

def login():

header["Referer"]="https://kyfw.12306.cn/otn/login/init"

r=reqSingle.get("https://kyfw.12306.cn/otn/login/init",verify=False,timeout=15,headers=header)

loginContent=r.content

aryKV=extractKey(loginContent,header)

#开始做验证码

while True:

r=reqSingle.get("https://kyfw.12306.cn/otn/passcodeNew/getPassCodeNew?module=login&rand=sjrand&",verify=False,timeout=5,headers=header)

withopen("loginRand.jpg","wb") as rimg:

rimg.write(r.content)

pass

img=Image.open("loginRand.jpg")

img.show()

randCode=raw_input("请输入登录验证码:")

#验证验证码

randReq= reqSingle.post("https://kyfw.12306.cn/otn/passcodeNew/checkRandCodeAnsyn",data={"rand":"sjrand","randCode":randCode},headers=header)

randRes=randReq.json()

if randRes.get("status")and randRes.get("httpstatus")==200 and randRes.get("data").get("result")=="1":

break;

pass

print "验证码输入正确!"

#开始登陆

loginRes=reqSingle.post("https://kyfw.12306.cn/otn/login/loginAysnSuggest",data={

aryKV[0]:aryKV[1],

"loginUserDTO.user_name":username,

"userDTO.password":password,

"randCode":randCode,

"myversion":"undefined",

"randCode_validate":""

},headers=header)

print repr(r.request)

print loginRes.content

loginResJson=loginRes.json()

if loginResJson.get("status")and loginResJson.get("httpstatus")==200:

if loginResJson.get("data")is not None and loginResJson.get("data").get("loginCheck")=="Y":

print "登录成功"

else:

raise Exception(loginRes.content)

else:

login()

pass

def checkTicket(dtStr):

print dt

while True:

try:

r= requests.get("https://kyfw.12306.cn/otn/leftTicket/queryT?leftTicketDTO.train_date="+dtStr+"&leftTicketDTO.from_station=BXP&leftTicketDTO.to_station=HDP&purpose_codes=ADULT",verify=False,timeout=5,headers=header)

break

except Exception:

pass

pass

#print r.contentfd

print r.content

try:

queryDataJson= json.loads(r.content)

except Exception:

return

if queryDataJson["httpstatus"]==200 and queryDataJson["status"] :

#print queryDataJson["data"]

for checiin queryDataJson["data"]:

tmpData=checi["queryLeftNewDTO"]

trainCode=tmpData.get("station_train_code")

#yzNum=tmpData.get("yz_num")

yzNum=tmpData.get("ze_num")

if trainCodein attCheCi:

if yzNum!="--" and yzNum!="无" and (yzNum=="有" or int(yzNum)>=2):

#发邮件

fromStation=tmpData.get("start_station_name")

toStation=tmpData.get("end_station_name")

secretStr=checi.get("secretStr")

orderTicket(fromStation, toStation, dtStr, secretStr)

#                     body=dtStr+"-"+trainCode+"-"+yzNum+u"个硬座"

#                     print body

#                     mailer=emailHelper.email_helper("smtp.qq.com", "fd", "fss", "qq.com","plain")

#                     mailer.send("630419595@qq.com", u"有火车票了",body)

#                     raise Exception("有票了")

pass

print trainCode+yzNum

pass

pass

pass

if __name__== '__main__':

#     login()

#     if checkIsLogin():

#         print "登录成功"

#

#     orderTicket("北京西","邯郸东","2015-01-14","MjAxNS0wMS0xNCMwMCNHNjczMSMwMjoxNSMwNzowNSMyNDAwMEc2NzMxMDUjQlhQI0hQUCMwOToyMCPljJfkuqzopb8j6YKv6YO45LicIzAxIzA2I08wMDAwMDA4MThNMDAwMDAwMTEwOTAwMDAwMDAyNiNQMiMxNDE5MDg2OTU2MTA0IzI5NEI0QkY0QTU2ODE2RDU1MzE5RkRCRkVEQzQ3Mzk2MUEyRUEwOEM0MUVCMjZGMDc3RUUyNzc0")

#     exit()

login()

if checkIsLogin():

print "登录成功"

while True:

checkCount=0

for dtin dateList:

checkTicket(dt)

time.sleep(2)

checkCount+=1

if checkCount%10==0:

if checkIsLogin():

print "成功状态"

else:

print "被踢了"

pass

pass

python12306买票_自己用python随意写的12306订票代码相关推荐

  1. 全国各地12306订票时间大集合

    全国各地12306订票时间大集合,12306什么时候放票一查便知道!本12306订票时间汇总是摘自12306官网最新时间表,买票之前,先看12306什么时候放票才能增加第一时间买票.想看各地12306 ...

  2. 12306订票抢票攻略:2013春节我要回家

    2019独角兽企业重金招聘Python工程师标准>>> 12306订票抢票攻略:2013春节我要回家 12306网上订火车票的人越来越多,但中国人多,不懂得12306网上订火车票官网 ...

  3. python12306买票_利用python代码写的12306订票代码

    本文实例讲述了python代码写的12306订票代码,分享给大家供大家参考. 具体实现方法如下: import datetime import json import re import sys im ...

  4. 12306订票候补是个坑_抢票神器成GitHub热榜第一,支持候补抢票,Python跑起来 | 标星8400...

    车栗子 发自 凹非寺  量子位 报道 | 公众号 QbitAI 想要十一回家的小伙伴们,大概刚刚经历了一波抢票大战. 顺便把一个Python抢票工具,送到了GitHub趋势榜第一: 项目名很干脆,就是 ...

  5. python+selenium实现12306抢票

    python+selenium实现12306抢票 一.准备工作 1.要先下载相关的包,selenium.interval.最好使用国内清华源 pip install (which package) - ...

  6. 可操作性强!Python实现一个电影订票系统!

    来源丨Python小二 一.效果展示 通过Python实现一个电影订票系统,效果如下所示: 二.整体结构图 三.代码分解 3.1 infos.py 一部电影的详细信息适合用 字典 结构来存储,我们可以 ...

  7. 用Python实现一个电影订票系统!

    一.效果展示  通过Python实现一个电影订票系统,效果如下所示: 二.整体结构图 三.代码分解 3.1 infos.py 一部电影的详细信息适合用 字典 结构来存储,我们可以给字典里添加多个键值对 ...

  8. 【文末赠书】用Python实现一个电影订票系统!

    来源丨Python小二 一.效果展示 通过Python实现一个电影订票系统,效果如下所示: 二.整体结构图 三.代码分解 3.1 infos.py 一部电影的详细信息适合用 字典 结构来存储,我们可以 ...

  9. 12306订票助手秘笈:2013春节我要回家

    12306订票助手秘笈:2013春节我要回家 又是一年春运时,一票难求让每个漂泊在外的异乡人更添几多思乡情.别急!12306抢票有技巧,本文帮你搜集了12306订票流程及订票技巧方面的文档资料,让我们 ...

最新文章

  1. windows nodejs mysql_windows server 安装 mysql + nondejs连接mysql
  2. c++多线程编程:常见面试题
  3. (二十六)、Java数组在内存中如何存放与分配
  4. mysql 获取自增id的值的方法
  5. 【五校联考6day2】yi
  6. 使用零代码平台构建应用,应该怎样转变思路?
  7. c语言程序设计 函数说课,《C语言程序设计》之函数说课课件.ppt
  8. error C3859: 超过了 PCH 的虚拟内存范围;请使用“-Zm276”或更大的命令行选项重新编译...
  9. EF DbContext.Configuration.ProxyCreationEnabled 什么鬼?
  10. IoTDB常用的SQL语句大全
  11. 充值校园卡显示服务器异常,调查| 为何这次校园卡系统故障时间这么长?
  12. 同时打开多个独立Excel窗口
  13. oom killer理解和日志分析
  14. [活动]2009年Linux内核开发者大会参与记录
  15. oracle 4098,ORA-04098错误解决方法-数据库专栏,ORACLE
  16. 录屏状态监听之防录屏 - iOS
  17. 官方免费的正版Xshell,人人都可以马上拥有
  18. Ubuntu怎么切换为中文版
  19. 监听器到底是什么,有什么用
  20. 安装envi时,步骤都正确,却还是停留在运行界面打不开可能是什么原因导致的

热门文章

  1. origin安装嵌入python_如何在 Origin 中切换镜像安装 Python 包
  2. 大型国企郑煤机集团引入电子签章,推动合同、单据、档案在线签
  3. Kanzi学习教程培训教程-Kanzi的简介和安装
  4. AM5728(AM5708)开发实战之修复AM5728不稳定问题
  5. swapidc的php语言,SWAPIDC目录说明及文件说明及某些常量内容
  6. 大型Android项目架构:基于组件化+模块化+Kotlin+协程+Flow+Retrofit+Jetpack+MVVM架构实现WanAndroid客户端
  7. DWM1000 测距原理简单分析 之 SS-TWR
  8. 化工专业学c语言吗,985院校的哪些专业在C+之下?
  9. 阿里云观察——阿里云总裁王坚专访
  10. hduoj 2034