作為搶課系統(tǒng)温艇,你必須要做到無論何時(shí)何地都在線捷雕,并且持續(xù)不斷的進(jìn)行搶課。只有這樣才能夠戰(zhàn)勝別的同學(xué)。
首先顾翼,這個(gè)搶課系統(tǒng)要做到自動(dòng)化屯伞,能夠斷線重連芜辕,其次是高速的選課律杠。
這里寫出的代碼是在沒有前端頁面的前提下臨時(shí)試用用的。所以很多方法的返回以及調(diào)用上不是很規(guī)范难菌。代碼中都有很詳細(xì)的注釋途乃,如果有不懂或者不住請(qǐng)務(wù)必告訴我。
登陸自動(dòng)化:
-
自動(dòng)輸入驗(yàn)證碼:
學(xué)校的選課系統(tǒng)登陸是需要驗(yàn)證碼的扔傅,經(jīng)過長達(dá)3天耍共,對(duì)學(xué)校js代碼的全部閱讀下(真的及其痛苦,期間遇到了無數(shù)的問題猎塞,看的差點(diǎn)吐出來)试读,我找到了學(xué)校驗(yàn)證碼的構(gòu)造方法,和登陸的傳輸方法荠耽。
下面是我的驗(yàn)證碼請(qǐng)求方法钩骇,返回的值為該次申請(qǐng)用的session以及一個(gè)可能的驗(yàn)證碼隊(duì)列
def get_vercode(url):
if len(os.listdir('static/vercode')) > 50: # 隔一段時(shí)間清理多余的驗(yàn)證碼圖片
for x in os.listdir('static/vercode'):
os.remove('static/vercode/'+x)
s = requests.session() # 確保申請(qǐng)驗(yàn)證碼的session和登陸時(shí)為一致的,所以寫在了這里
# 構(gòu)建驗(yàn)證碼路徑
now_time = str(int(time.time()))
pic_url = url + 'servlet/ImageServlet?d=' + now_time # 學(xué)校的驗(yàn)證碼就是這樣,js獲取時(shí)間后先服務(wù)器索要一個(gè)驗(yàn)證碼圖片
# 取得驗(yàn)證碼圖片
pic = s.get(pic_url).content
filename = 'static/vercode/' + now_time + '.jpg' # 以后可能還是要用到手動(dòng)輸入驗(yàn)證碼倘屹,所以先保存圖片吧
with open(filename, 'wb') as f:
f.write(pic)
img = Image.open(filename)
vcode = pytesseract.image_to_string(img) # 使用ocr技術(shù)將圖片中的驗(yàn)證碼讀取出來
time.sleep(0.3) # 休息一會(huì)银亲,等待驗(yàn)證碼完全讀取完成(我在電腦上有時(shí)候讀取不完。纽匙。
vcode_list = confirm_vcode(vcode) # 檢驗(yàn)驗(yàn)證碼有效性务蝠,并輸出更好的驗(yàn)證碼隊(duì)列
return s, vcode_list
-
驗(yàn)證碼自動(dòng)識(shí)別正確性檢驗(yàn):
使用pytesseract生成的驗(yàn)證碼往往有著這樣那樣的錯(cuò)誤,所以我要對(duì)返回的驗(yàn)證碼進(jìn)行一些小小的調(diào)教烛缔,以提高效率以及降低資源的占用
def confirm_vcode(vcode):
# 這個(gè)函數(shù)用來驗(yàn)證自動(dòng)生成的驗(yàn)證碼的有效性馏段,并試圖優(yōu)化它
# 學(xué)校的驗(yàn)證碼會(huì)忽略零和o的區(qū)別,都當(dāng)做同一個(gè)字符践瓷,但是這個(gè)ocr總是會(huì)把5和S搞混
if len(vcode) != 4: # 不是4位肯定錯(cuò)了院喜,直接跳過吧
return False
vcode_list = []
vcode_list.append(vcode)
vcode_list.append(vcode.replace('S', '5')) # 置換S和5,然后存入數(shù)列中
vcode_list.append(vcode.replace('5', 'S'))
return vcode_list
-
登陸主體部分:
這里因?yàn)榍芭_(tái)沒搭建好晕翠,一時(shí)不知道該返回什么喷舀,目前只是做了一個(gè)登陸而已
def login(url, data): # 未來可能要支持不同的學(xué)校,url先給出來
i = 0
while i == 0:
info = get_vercode(url)
s = info[0] # 將前面獲取驗(yàn)證碼用的session賦予s
if info[1] is False: # 如果這次驗(yàn)證碼有問題就重新取一次驗(yàn)證碼
continue
for x in info[1]: # 遍歷驗(yàn)證碼列表
data['verifyCode'] = x
r = s.post(url+'loadData.xk?method=checkLogin', data) # 傳輸?shù)顷憯?shù)據(jù)淋肾,data是外部給的元咙,這里寫出來也沒啥意義了
print(r.text)
if r.text == '{"success":true,"message":""}': # 成功登陸了就準(zhǔn)備跳出登陸循環(huán)吧
print('成功了')
i += 1
break
選課:
就很簡(jiǎn)單了,略過吧
遇到的問題:
無法獲取到教學(xué)計(jì)劃和本學(xué)期待選課表∥自保看了三天代碼都沒有找到這些數(shù)據(jù)是在哪里傳輸?shù)摹U娴暮鼙罎⒓坠鳌5侨绻媚M瀏覽器的方法會(huì)占用大量的資源简识,為了能夠很好的在廉價(jià)的服務(wù)器上發(fā)布,這種方法還是先不考慮了感猛。先做前臺(tái)頁面七扰,然后再進(jìn)行這方面的研究吧