最近在學Python,因此用Python寫了這個12306搶票腳本,分享出來,與你們共同交流和學習,有不對的地方,請你們多多指正。話很少說,進入正題:php
在進入正題以前,我想說明一下,因爲12306官網的改版更新,因此腳本做了一點小小的變化,具體修改後的源碼,能夠到GitHub上面查看……新版腳本源碼css
這個腳本目前只能刷一趟車的,人數能夠是多個,支持選取做爲類型等。
實現思路是splinter.browser模擬瀏覽器登錄和操做,因爲12306的驗證碼很差自動識別,因此,驗證碼須要用戶進行手動識別,並進行登錄操做,以後的事情,就交由腳原本操做就能夠了,下面是我測試時候的一些截圖:html
第一步:以下圖,首先輸入搶票基本信息python
第二步:而後進入登陸頁,須要手動輸入驗證碼,並點擊登錄操做
git
第三步:登錄後,自動進入到搶票頁面,以下圖這樣的
github
最後:就是坐等刷票結果就行了,以下圖這樣,就說是刷票成功了,刷到票後,會進行短信和郵件的通知,請記得及時前往12306進行支付,否則就白搶了。
web
Python運行環境:python3.6
用到的模塊:re、splinter、time、sys、httplib二、urllib、smtplib、email
未安裝的模塊,請使用pip instatll進行安裝,例如:pip install splinter
以下代碼是這個腳本全部用到的模塊引入:
chrome
import re
from splinter.browser import Browser
from time import sleep
import sys
import httplib2
from urllib import parse
import smtplib
from email.mime.text import MIMEText
複製代碼
刷票前信息準備,我主要說一下始發站和目的地的cookie值獲取,由於輸入城市的時候,須要經過cookie值,cookie值能夠經過12306官網,而後在F12(相信全部的coder都知道這個吧)的network裏面的查詢請求cookie中能夠看到,在請求的header裏面能夠找到,_jc_save_fromStation值是出發站的cookie,_jc_save_toStation的值是目的地的cookie,而後加入到代碼裏的城市的cookie字典city_list裏便可,鍵是城市的首字母,值是cookie值的形式。json
搶票,確定須要先登陸,我這裏模擬的登陸操做,會自動填充12306的帳號名和密碼,固然,你也能夠在打開的瀏覽器中修改帳號和密碼,實現的關鍵代碼以下:api
def do_login(self):
"""登陸功能實現,手動識別驗證碼進行登陸"""
self.driver.visit(self.login_url)
sleep(1)
self.driver.fill('loginUserDTO.user_name', self.user_name)
self.driver.fill('userDTO.password', self.password)
print('請輸入驗證碼……')
while True:
if self.driver.url != self.init_my_url:
sleep(1)
else:
break
複製代碼
登陸以後,就是控制刷票的各類操做處理了,這裏,我就不貼代碼了,由於代碼比較多,別擔憂,在最後,我會貼出完整的代碼的。
當刷票成功後,我會進行短信和郵件的雙重通知,固然,這裏短信通知的平臺,就看你用那個具體來修改代碼了,我用的是互億無線的體驗版的免費短信通知接口;發送郵件模塊我用的是smtplib,發送郵件服務器用的是163郵箱,若是用163郵箱的話,你尚未設置客戶端受權密碼,記得先設置客戶端受權密碼就行了,挺方便的。如下是主要實現代碼:
def send_sms(self, mobile, sms_info):
"""發送手機通知短信,用的是-互億無線-的測試短信"""
host = "106.ihuyi.com"
sms_send_uri = "/webservice/sms.php?method=Submit"
account = "C59782899"
pass_word = "19d4d9c0796532c7328e8b82e2812655"
params = parse.urlencode(
{'account': account, 'password': pass_word, 'content': sms_info, 'mobile': mobile, 'format': 'json'}
)
headers = {"Content-type": "application/x-www-form-urlencoded", "Accept": "text/plain"}
conn = httplib2.HTTPConnectionWithTimeout(host, port=80, timeout=30)
conn.request("POST", sms_send_uri, params, headers)
response = conn.getresponse()
response_str = response.read()
conn.close()
return response_str
def send_mail(self, receiver_address, content):
"""發送郵件通知"""
# 鏈接郵箱服務器信息
host = 'smtp.163.com'
port = 25
sender = 'xxxxxx@163.com' # 你的發件郵箱號碼
pwd = '******' # 不是登錄密碼,是客戶端受權密碼
# 發件信息
receiver = receiver_address
body = '<h2>舒適提醒:</h2><p>' + content + '</p>'
msg = MIMEText(body, 'html', _charset="utf-8")
msg['subject'] = '搶票成功通知!'
msg['from'] = sender
msg['to'] = receiver
s = smtplib.SMTP(host, port)
# 開始登錄郵箱,併發送郵件
s.login(sender, pwd)
s.sendmail(sender, receiver, msg.as_string())
複製代碼
說了那麼多,感受都是說了好多廢話啊,哈哈,很差意思,耽誤你們時間來看我瞎扯了,我貼上你們最關心的源碼,請接碼,你們在嘗試運行過程當中,有任何問題,能夠給我留言或者私信我,我看到都會及時回覆你們的:
#!/usr/bin/env python
# -*- coding: utf-8 -*-
""" 經過splinter刷12306火車票 能夠自動填充帳號密碼,同時,在登陸時,也能夠修改帳號密碼 而後手動識別驗證碼,並登錄,接下來的事情,交由腳原本作了,靜靜的等待搶票結果就好(刷票過程當中,瀏覽器不可關閉) author: cuizy time: 2018-05-30 """
import re
from splinter.browser import Browser
from time import sleep
import sys
import httplib2
from urllib import parse
import smtplib
from email.mime.text import MIMEText
class BrushTicket(object):
"""買票類及實現方法"""
def __init__(self, user_name, password, passengers, from_time, from_station, to_station, number, seat_type, receiver_mobile, receiver_email):
"""定義實例屬性,初始化"""
# 1206帳號密碼
self.user_name = user_name
self.password = password
# 乘客姓名
self.passengers = passengers
# 起始站和終點站
self.from_station = from_station
self.to_station = to_station
# 乘車日期
self.from_time = from_time
# 車次編號
self.number = number.capitalize()
# 座位類型所在td位置
if seat_type == '商務座特等座':
seat_type_index = 1
seat_type_value = 9
elif seat_type == '一等座':
seat_type_index = 2
seat_type_value = 'M'
elif seat_type == '二等座':
seat_type_index = 3
seat_type_value = 0
elif seat_type == '高級軟臥':
seat_type_index = 4
seat_type_value = 6
elif seat_type == '軟臥':
seat_type_index = 5
seat_type_value = 4
elif seat_type == '動臥':
seat_type_index = 6
seat_type_value = 'F'
elif seat_type == '硬臥':
seat_type_index = 7
seat_type_value = 3
elif seat_type == '軟座':
seat_type_index = 8
seat_type_value = 2
elif seat_type == '硬座':
seat_type_index = 9
seat_type_value = 1
elif seat_type == '無座':
seat_type_index = 10
seat_type_value = 1
elif seat_type == '其餘':
seat_type_index = 11
seat_type_value = 1
else:
seat_type_index = 7
seat_type_value = 3
self.seat_type_index = seat_type_index
self.seat_type_value = seat_type_value
# 通知信息
self.receiver_mobile = receiver_mobile
self.receiver_email = receiver_email
# 主要頁面網址
self.login_url = 'https://kyfw.12306.cn/otn/login/init'
self.init_my_url = 'https://kyfw.12306.cn/otn/index/initMy12306'
self.ticket_url = 'https://kyfw.12306.cn/otn/leftTicket/init'
# 瀏覽器驅動信息,驅動下載頁:https://sites.google.com/a/chromium.org/chromedriver/downloads
self.driver_name = 'chrome'
self.executable_path = 'C:\\Users\cuizy\AppData\Local\Programs\Python\Python36\Scripts\chromedriver.exe'
def do_login(self):
"""登陸功能實現,手動識別驗證碼進行登陸"""
self.driver.visit(self.login_url)
sleep(1)
self.driver.fill('loginUserDTO.user_name', self.user_name)
self.driver.fill('userDTO.password', self.password)
print('請輸入驗證碼……')
while True:
if self.driver.url != self.init_my_url:
sleep(1)
else:
break
def start_brush(self):
"""買票功能實現"""
self.driver = Browser(driver_name=self.driver_name, executable_path=self.executable_path)
# 瀏覽器窗口的大小
self.driver.driver.set_window_size(900, 700)
self.do_login()
self.driver.visit(self.ticket_url)
try:
print('開始刷票……')
# 加載車票查詢信息
self.driver.cookies.add({"_jc_save_fromStation": self.from_station})
self.driver.cookies.add({"_jc_save_toStation": self.to_station})
self.driver.cookies.add({"_jc_save_fromDate": self.from_time})
self.driver.reload()
count = 0
while self.driver.url.split('?')[0] == self.ticket_url:
self.driver.find_by_text('查詢').click()
sleep(1)
count += 1
print('第%d次點擊查詢……' % count)
try:
car_no_location = self.driver.find_by_id("queryLeftTable")[0].find_by_text(self.number)[1]
current_tr = car_no_location.find_by_xpath("./../../../../..")
if current_tr.find_by_tag('td')[self.seat_type_index].text == '--':
print('無此座位類型出售,已結束當前刷票,請從新開啓!')
sys.exit(1)
elif current_tr.find_by_tag('td')[self.seat_type_index].text == '無':
print('無票,繼續嘗試……')
else:
# 有票,嘗試預訂
print('刷到票了(餘票數:' + str(current_tr.find_by_tag('td')[self.seat_type_index].text) + '),開始嘗試預訂……')
current_tr.find_by_css('td.no-br>a')[0].click()
sleep(1)
key_value = 1
for p in self.passengers:
# 選擇用戶
print('開始選擇用戶……')
self.driver.find_by_text(p).last.click()
# 選擇座位類型
print('開始選擇席別……')
if self.seat_type_value != 0:
seat_select = self.driver.find_by_id("seatType_" + str(key_value))[0]
seat_select.find_by_xpath("//option[@value='" + str(self.seat_type_value) + "']")[0].click()
key_value += 1
sleep(0.5)
if p[-1] == ')':
self.driver.find_by_id('dialog_xsertcj_ok').click()
print('正在提交訂單……')
self.driver.find_by_id('submitOrder_id').click()
sleep(2)
# 查看放回結果是否正常
submit_false_info = self.driver.find_by_id('orderResultInfo_id')[0].text
if submit_false_info != '':
print(submit_false_info)
self.driver.find_by_id('qr_closeTranforDialog_id').click()
sleep(0.2)
self.driver.find_by_id('preStep_id').click()
sleep(0.3)
continue
print('正在確認訂單……')
self.driver.find_by_id('qr_submit_id').click()
print('預訂成功,請及時前往支付……')
# 發送通知信息
self.send_mail(self.receiver_email, '恭喜您,搶到票了,請及時前往12306支付訂單!')
self.send_sms(self.receiver_mobile, '您的驗證碼是:8888。請不要把驗證碼泄露給其餘人。')
except Exception as error_info:
print(error_info)
except Exception as error_info:
print(error_info)
def send_sms(self, mobile, sms_info):
"""發送手機通知短信,用的是-互億無線-的測試短信"""
host = "106.ihuyi.com"
sms_send_uri = "/webservice/sms.php?method=Submit"
account = "C59782899"
pass_word = "19d4d9c0796532c7328e8b82e2812655"
params = parse.urlencode(
{'account': account, 'password': pass_word, 'content': sms_info, 'mobile': mobile, 'format': 'json'}
)
headers = {"Content-type": "application/x-www-form-urlencoded", "Accept": "text/plain"}
conn = httplib2.HTTPConnectionWithTimeout(host, port=80, timeout=30)
conn.request("POST", sms_send_uri, params, headers)
response = conn.getresponse()
response_str = response.read()
conn.close()
return response_str
def send_mail(self, receiver_address, content):
"""發送郵件通知"""
# 鏈接郵箱服務器信息
host = 'smtp.163.com'
port = 25
sender = '******@163.com' # 你的發件郵箱號碼
pwd = '******' # 不是登錄密碼,是客戶端受權密碼
# 發件信息
receiver = receiver_address
body = '<h2>舒適提醒:</h2><p>' + content + '</p>'
msg = MIMEText(body, 'html', _charset="utf-8")
msg['subject'] = '搶票成功通知!'
msg['from'] = sender
msg['to'] = receiver
s = smtplib.SMTP(host, port)
# 開始登錄郵箱,併發送郵件
s.login(sender, pwd)
s.sendmail(sender, receiver, msg.as_string())
if __name__ == '__main__':
# 12306用戶名
user_name = input('請輸入12306用戶名:')
while user_name == '':
user_name = input('12306用戶名不能爲空,請從新輸入:')
# 12306登錄密碼
password = input('請輸入12306登錄密碼:')
while password == '':
password = input('12306登錄密碼不能爲空,請從新輸入:')
# 乘客姓名
passengers_input = input('請輸入乘車人姓名,多人用英文逗號「,」鏈接,(例如單人「張三」或者多人「張三,李四」):')
passengers = passengers_input.split(",")
while passengers_input == '' or len(passengers) > 4:
print('乘車人最少1位,最多4位!')
passengers_input = input('請從新輸入乘車人姓名,多人用英文逗號「,」鏈接,(例如單人「張三」或者多人「張三,李四」):')
passengers = passengers_input.split(",")
# 乘車日期
from_time = input('請輸入乘車日期(例如「2018-08-08」):')
date_pattern = re.compile(r'^\d{4}-\d{2}-\d{2}$')
while from_time == '' or re.findall(date_pattern, from_time) == []:
from_time = input('乘車日期不能爲空或者時間格式不正確,請從新輸入:')
# 城市cookie字典
city_list = {
'bj': '%u5317%u4EAC%2CBJP', # 北京
'hd': '%u5929%u6D25%2CTJP', # 邯鄲
'nn': '%u5357%u5B81%2CNNZ', # 南寧
'wh': '%u6B66%u6C49%2CWHN', # 武漢
'cs': '%u957F%u6C99%2CCSQ', # 長沙
'ty': '%u592A%u539F%2CTYV', # 太原
'yc': '%u8FD0%u57CE%2CYNV', # 運城
'gzn': '%u5E7F%u5DDE%u5357%2CIZQ', # 廣州南
'wzn': '%u68A7%u5DDE%u5357%2CWBZ', # 梧州南
}
# 出發站
from_input = input('請輸入出發站,只須要輸入首字母就行(例如北京「bj」):')
while from_input not in city_list.keys():
from_input = input('出發站不能爲空或不支持當前出發站(若有須要,請聯繫管理員!),請從新輸入:')
from_station = city_list[from_input]
# 終點站
to_input = input('請輸入終點站,只須要輸入首字母就行(例如北京「bj」):')
while to_input not in city_list.keys():
to_input = input('終點站不能爲空或不支持當前終點站(若有須要,請聯繫管理員!),請從新輸入:')
to_station = city_list[to_input]
# 車次編號
number = input('請輸入車次號(例如「G110」):')
while number == '':
number = input('車次號不能爲空,請從新輸入:')
# 座位類型
seat_type = input('請輸入座位類型(例如「軟臥」):')
while seat_type == '':
seat_type = input('座位類型不能爲空,請從新輸入:')
# 搶票成功,通知該手機號碼
receiver_mobile = input('請預留一個手機號碼,方便搶到票後進行通知(例如:18888888888):')
mobile_pattern = re.compile(r'^1{1}\d{10}$')
while receiver_mobile == '' or re.findall(mobile_pattern, receiver_mobile) == []:
receiver_mobile = input('預留手機號碼不能爲空或者格式不正確,請從新輸入:')
receiver_email = input('請預留一個郵箱,方便搶到票後進行通知(例如:test@163.com):')
while receiver_email == '':
receiver_email = input('預留郵箱不能爲空,請從新輸入:')
# 開始搶票
ticket = BrushTicket(user_name, password, passengers, from_time, from_station, to_station, number, seat_type, receiver_mobile, receiver_email)
ticket.start_brush()
複製代碼