-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathAutoticket.py
executable file
·445 lines (397 loc) · 21 KB
/
Autoticket.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
# coding: utf-8
from json import loads
from os.path import exists
from pickle import dump, load
from time import sleep, time
from selenium import webdriver
from selenium.webdriver.common.by import By
from selenium.webdriver.support.wait import WebDriverWait
from selenium.webdriver.support import expected_conditions as EC
# 倒入配置文件
from config import FLAG
from config import LOG
# 初始化log
log=LOG('auto_ticket.log',level='debug')
class Concert(object):
def __init__(self, session, price, date, real_name, nick_name, ticket_num, damai_url, target_url, browser):
self.session = session # 场次序号优先级
self.price = price # 票价序号优先级
self.date = date # 日期选择
self.real_name = real_name # 实名者序号
self.status = 0 # 状态标记
self.time_start = 0 # 开始时间
self.time_end = 0 # 结束时间
self.num = 0 # 尝试次数
self.type = 0 # 目标购票网址类别
self.ticket_num = ticket_num # 购买票数
self.nick_name = nick_name # 用户昵称
self.damai_url = damai_url # 大麦网官网网址
self.target_url = target_url # 目标购票网址
self.browser = browser # 只设置了chrome
self.total_wait_time = 3 # 页面元素加载总等待时间
self.refresh_wait_time = 0.3 # 页面元素等待刷新时间
self.intersect_wait_time = 0.5 # 间隔等待时间,防止速度过快导致问题
if self.target_url.find("detail.damai.cn") != -1:
self.type = 1
elif self.target_url.find("piao.damai.cn") != -1:
self.type = 2
else:
self.type = 0
self.driver.quit()
raise Exception("***Error:Unsupported Target Url Format:{}***".format(self.target_url))
def isClassPresent(self, item, name, ret=False):
try:
result = item.find_element_by_class_name(name)
if ret:
return result
else:
return True
except:
return False
def get_cookie(self):
self.driver.get(self.damai_url)
log.logger.info("###请点击登录###")
while self.driver.title.find('大麦网-全球演出赛事官方购票平台') != -1: # 等待网页加载完成
sleep(1)
log.logger.info("###请扫码登录###")
while self.driver.title == '大麦登录': # 等待扫码完成
sleep(1)
dump(self.driver.get_cookies(), open("cookies.pkl", "wb"))
log.logger.info("###Cookie保存成功###")
def set_cookie(self):
try:
cookies = load(open("cookies.pkl", "rb")) # 载入cookie
for cookie in cookies:
cookie_dict = {
'domain': '.damai.cn', # 必须有,不然就是假登录
'name': cookie.get('name'),
'value': cookie.get('value'),
"expires": "",
'path': '/',
'httpOnly': False,
'HostOnly': False,
'Secure': False}
self.driver.add_cookie(cookie_dict)
log.logger.info('###载入Cookie###')
except Exception as e:
print(e)
def login(self):
if not exists('cookies.pkl'): # 如果不存在cookie.pkl,就获取一下
if self.browser == 'Chrome': # 选择了Chrome浏览器
self.driver = webdriver.Chrome('./chromedriver')
elif self.browser == 1: # 选择了Firefox浏览器
self.driver = webdriver.Firefox()
else:
log.logger.error("***错误:未知的浏览器类别***")
raise Exception("***错误:未知的浏览器类别***")
self.get_cookie()
self.driver.quit()
log.logger.info('###打开浏览器,进入大麦网###')
if self.browser == 'Chrome': # 选择了Chrome浏览器,并成功加载cookie,设置不载入图片,提高刷新效率
options = webdriver.ChromeOptions()
prefs = {"profile.managed_default_content_settings.images":2}
options.add_experimental_option("prefs",prefs)
self.driver = webdriver.Chrome('./chromedriver')
elif self.browser == 1: # 选择了火狐浏览器
options = webdriver.FirefoxProfile()
options.set_preference('permissions.default.image', 2)
self.driver = webdriver.Firefox(options)
else:
log.logger.error("***错误:未知的浏览器类别***")
raise Exception("***错误:未知的浏览器类别***")
self.driver.get(self.target_url)
self.set_cookie()
# self.driver.maximize_window()
self.driver.refresh()
def enter_concert(self):
self.login()
try:
if self.type == 1: # detail.damai.cn
locator = (By.XPATH, "/html/body/div[1]/div/div[3]/div[1]/a[2]/div")
elif self.type == 2: # piao.damai.cn
locator = (By.XPATH, "/html/body/div[1]/div/ul/li[2]/div/label/a[2]")
WebDriverWait(self.driver, self.total_wait_time, self.refresh_wait_time).until(
EC.text_to_be_present_in_element(locator, self.nick_name))
self.status = 1
log.logger.info("###登录成功###")
except Exception as e:
log.logger.error("***错误:登录失败,请检查配置文件昵称或删除cookie.pkl后重试***")
self.status = 0
self.driver.quit()
raise Exception("***错误:登录失败,请检查配置文件昵称或删除cookie.pkl后重试***")
def choose_ticket_1(self): # for type 1, i.e., detail.damai.cn
self.time_start = time()
log.logger.info("###开始进行日期及票价选择###")
while self.driver.title.find('确认订单') == -1: # 如果跳转到了确认界面就算这步成功了,否则继续执行此步
self.num += 1 # 记录抢票轮数
if self.date != 0: # 如果需要选择日期
calendar = WebDriverWait(self.driver, self.total_wait_time, self.refresh_wait_time).until(
EC.presence_of_element_located((By.CLASS_NAME, "functional-calendar")))
datelist = calendar.find_elements_by_css_selector("[class='wh_content_item']") # 找到能选择的日期
datelist = datelist[7:] # 跳过前面7个表示周一~周日的元素
datelist[self.date - 1].click() # 选择对应日期
selects = self.driver.find_elements_by_class_name('perform__order__select')
# print('可选区域数量为:{}'.format(len(selects)))
for item in selects:
if item.find_element_by_class_name('select_left').text == '场次':
session = item
# print('\t场次定位成功')
elif item.find_element_by_class_name('select_left').text == '票档':
price = item
# print('\t票档定位成功')
session_list = session.find_elements_by_class_name('select_right_list_item')
log.logger.info('可选场次数量为:{}'.format(len(session_list)))
if len(self.session) == 1:
j = session_list[self.session[0] - 1].click()
else:
for i in self.session: # 根据优先级选择一个可行场次
j = session_list[i - 1]
k = self.isClassPresent(j, 'presell', True)
if k: # 如果找到了带presell的类
if k.text == '无票':
continue
elif k.text == '预售':
j.click()
break
else:
j.click()
break
price_list = price.find_elements_by_class_name('select_right_list_item')
log.logger.info('可选票档数量为:{}'.format(len(price_list)))
if len(self.price) == 1:
j = price_list[self.price[0] - 1].click()
else:
for i in self.price:
j = price_list[i - 1]
print(j)
k = self.isClassPresent(j, 'notticket')
if k: # 存在notticket代表存在缺货登记,跳过
continue
else:
j.click()
continue
buybutton = self.driver.find_element_by_class_name('buybtn')
buybutton_text = buybutton.text
# print(buybutton_text)
def add_ticket(): # 设置增加票数
try:
for i in range(self.ticket_num - 1):
addbtn = WebDriverWait(self.driver, self.total_wait_time, self.refresh_wait_time).until(
EC.presence_of_element_located((By.XPATH, "//div[@class='cafe-c-input-number']/a[2]")))
addbtn.click()
except:
log.logger.error("***错误:票数增加失败***")
raise Exception("***错误:票数增加失败***")
if buybutton_text == "即将开抢" or buybutton_text == "即将开售":
self.status = 2
self.driver.refresh()
log.logger.warn('---尚未开售,刷新等待---')
continue
elif buybutton_text == "立即预订":
add_ticket()
buybutton.click()
self.status = 3
elif buybutton_text == "立即购买":
add_ticket()
buybutton.click()
self.status = 4
elif buybutton_text == "选座购买": # 选座购买暂时无法完成自动化
# buybutton.click()
self.status = 5
print("###请自行选择位置和票价###")
break
elif buybutton_text == "提交缺货登记":
log.logger.warn('###抢票失败,请手动提交缺货登记###')
break
def choose_ticket_2(self): # for type 2, i.e., piao.damai.cn
self.time_start = time()
log.logger.info("###开始进行日期及票价选择###")
while self.driver.title.find('订单结算页') == -1: # 如果跳转到了确认界面就算这步成功了,否则继续执行此步
self.num += 1 # 记录抢票轮数
if self.date != 0: # 如果要选择日期
datepicker = WebDriverWait(self.driver, self.total_wait_time, self.refresh_wait_time).until(
EC.presence_of_element_located(
(By.CLASS_NAME, "month")))
datelist = datepicker.find_elements_by_tag_name("span") # 找出所有日期
# print(len(datelist))
validlist = []
for i in range(len(datelist)): # 筛选出所有可选择日期
j = datelist[i]
k = j.get_attribute('class')
if k == 'itm z-show itm-undefined z-sel' \
or k == 'itm z-show itm-undefined' \
or k == 'itm itm-end z-show itm-undefined':
validlist.append(j)
# print(len(validlist))
validlist[self.date - 1].click()
session = WebDriverWait(self.driver, self.total_wait_time, self.refresh_wait_time).until(
EC.presence_of_element_located(
(By.ID, "performList")))
# session = self.driver.find_element_by_id('performList')
session_list = session.find_elements_by_tag_name('li')
log.logger.info('可选场次数量为:{}'.format(len(session_list)))
for i in self.session: # 根据优先级选择一个可行场次,目前暂时没有找到有不可行日期的案例
j = session_list[i - 1]
k = j.get_attribute('class').strip()
if k == 'itm' or k == 'itm j_more': # 未选中
j.find_element_by_tag_name('a').click()
break
elif k == 'itm itm-sel' or k == 'itm j_more itm-sel': # 已选中
break
elif k == 'itm itm-oos': # 无法选中
continue
sleep(self.intersect_wait_time)
price = WebDriverWait(self.driver, self.total_wait_time, self.refresh_wait_time).until(
EC.presence_of_element_located(
(By.ID, "priceList")))
# price = self.driver.find_element_by_id('priceList')
price_list = price.find_elements_by_tag_name('li')
log.logger.log('可选票档数量为:{}'.format(len(price_list)))
for i in self.price:
j = price_list[i - 1]
k = j.get_attribute('class').strip()
if k == 'itm' or k == 'itm j_more': # 未选中
j.find_element_by_tag_name('a').click()
break
elif k == 'itm itm-sel' or k == 'itm j_more itm-sel': # 已选中
break
elif k == 'itm itm-oos': # 无法选中
continue
buybutton = None
try:
buybutton = self.driver.find_element_by_id('btnBuyNow') # 要改成立即预订按钮的id
self.status = 3
except:
try:
buybutton = self.driver.find_element_by_id('btnBuyNow')
self.status = 4
except:
log.logger.info('###无法立即购买,尝试自行选座###')
try:
buybutton = self.driver.find_element_by_id('btnXuanzuo')
self.status = 5
log.logger.info("###请自行选择位置和票价###")
break
except:
log.logger.info('---尚未开售,刷新等待---')
self.status = 2
self.driver.refresh()
# 需要先判断是否存在按钮,才能确定是否会出现添加票
if self.ticket_num > 1 and self.status not in [2, 5]: # 自动添加购票数
# add = self.driver.find_element_by_class_name('btn btn-add')
add = WebDriverWait(self.driver, self.total_wait_time, self.refresh_wait_time).until(
EC.presence_of_element_located(
(By.CLASS_NAME, "btn-add")))
for i in range(self.ticket_num - 1):
add.click()
buybutton.click()
# 目前没有找到缺货没有按钮的情况
def check_order_1(self):
if self.status in [3, 4]:
log.logger.info('###开始确认订单###')
button_xpath = " //*[@id=\"confirmOrder_1\"]/div[%d]/button" # 同意以上协议并提交订单Xpath
button_replace = 8 # 当实名者信息不空时为9,空时为8
if self.real_name: # 实名者信息不为空
button_replace = 9
log.logger.info('###选择购票人信息###')
try:
list_xpath = "//*[@id=\"confirmOrder_1\"]/div[2]/div[2]/div[1]/div[%d]/label/span[1]/input"
for i in range(len(self.real_name)): # 选择第i个实名者
WebDriverWait(self.driver, self.total_wait_time, self.refresh_wait_time).until(
EC.presence_of_element_located((By.XPATH, list_xpath%(i+1)))).click()
except Exception as e:
log.logger.error("***错误:实名信息框未显示,请检查网络或配置文件***")
raise Exception("***错误:实名信息框未显示,请检查网络或配置文件***")
submitbtn = WebDriverWait(self.driver, self.total_wait_time, self.refresh_wait_time).until(
EC.presence_of_element_located(
(By.XPATH, button_xpath%button_replace))) # 同意以上协议并提交订单
submitbtn.click()
'''# 以下的方法更通用,但是更慢
try:
buttons = self.driver.find_elements_by_tag_name('button') # 找出所有该页面的button
for button in buttons:
if button.text == '同意以上协议并提交订单':
button.click()
break
except Exception as e:
raise Exception('***错误:没有找到提交订单按钮***')
'''
try:
WebDriverWait(self.driver, self.total_wait_time, self.refresh_wait_time).until(
EC.title_contains('支付宝'))
self.status = 6
log.logger.warn('###成功提交订单,请手动支付###')
self.time_end = time()
except Exception as e:
log.logger.warn('---提交订单失败,请查看问题---')
log.logger.error(e)
def check_order_2(self):
if self.status in [3, 4]:
log.logger.info('###开始确认订单###')
if self.real_name: # 实名者信息不为空
log.logger.info('###选择购票人信息###')
try:
tb = WebDriverWait(self.driver, self.total_wait_time, self.refresh_wait_time).until(
EC.presence_of_element_located(
(By.CLASS_NAME, 'from-1')))
tb.find_element_by_tag_name('a').click() # 点击选择购票人按钮
sleep(self.intersect_wait_time)
# 此处好像定位不到实名者框,还没有解决
lb_list = WebDriverWait(self.driver, self.total_wait_time, self.refresh_wait_time).until(
EC.presence_of_element_located(
(By.XPATH, '/html/body/div[3]/div[3]/div[12]/div/div[2]/div/div[2]/div/table/tbody'))) # 定位弹窗
lb = lb_list.find_elements_by_tag_name('input')
for i in range(len(self.real_name)):
lb[self.real_name[i] - 1].find_element_by_tag_name('input').click() # 选择第self.real_name个实名者
except Exception as e:
print(e)
input('halt')
WebDriverWait(self.driver, self.total_wait_time, self.refresh_wait_time).until(
EC.presence_of_element_located(
(By.ID, 'orderConfirmSubmit'))).click() # 同意以上协议并提交订单
# self.driver.find_element_by_id('orderConfirmSubmit').click()
element = WebDriverWait(self.driver, 10, self.refresh_wait_time).until(EC.title_contains('选择支付方式'))
element.find_element_by_xpath('/html/body/div[5]/div/div/div/ul/li[2]/a').click() # 默认选择支付宝
element.find_element_by_xpath('/html/body/div[5]/div/div/form/div[2]/ul/li[1]/label/input').click()
element.find_element_by_id('submit2').click() # 确认无误,支付
self.status = 6
log.logger.info('###成功提交订单,请手动支付###')
self.time_end = time()
# print('###提交订单失败,请查看问题###') # 这里异常处理还有点问题
def finish(self):
if self.status == 6: # 说明抢票成功
log.logger.log("###经过%d轮奋斗,共耗时%f秒,抢票成功!请确认订单信息###" % (self.num, round(self.time_end - self.time_start, 3)))
else:
self.driver.quit()
def run():
log.logger.info('hello')
try:
con = Concert(FLAG.sess,
FLAG.price,
FLAG.date,
FLAG.real_name,
FLAG.nick_name,
FLAG.ticket_num,
FLAG.base_url,
FLAG.target_url,
FLAG.browse)
except Exception as e:
log.logger.error("***错误:初始化失败,请检查配置文件***")
raise Exception("***错误:初始化失败,请检查配置文件***")
con.enter_concert()
while True: # 可用于无限抢票,防止弹窗类异常使抢票终止
if True:
try:
if con.type == 1: # detail.damai.cn
con.choose_ticket_1()
con.check_order_1()
elif con.type == 2: # piao.damai.cn
con.choose_ticket_2()
con.check_order_2()
break
except Exception as e:
log.logger.error(e)
#con.driver.get(con.target_url)
con.finish()
if __name__ == '__main__':
run()