事情是這樣子的,昨天有粉絲私信問我能不能寫個腳本搶冰墩墩:
我尋思著這好像也不難吧纸泡,不就是寫一個淘寶搶購的腳本么漂问,于是加班加點寫了一個,結(jié)果寫完測試的時候發(fā)現(xiàn)官方的冰墩墩現(xiàn)在已經(jīng)下架了,只有一些非官方的店在售賣蚤假,也不知道貨是真的還是假的栏饮。不過腳本既然已經(jīng)寫完了,那還是上來分享一波吧磷仰。萬一重新上架了大家還是可以用這個腳本搶購的袍嬉,雖然我仔細看了一下,這吉祥物并不是我喜歡的風(fēng)格芒划,不然我肯定先自己搶到了再上來分享這個腳本冬竟。
廢話不多說,讓我們愉快地開始吧~
相關(guān)文件
需要源碼的可以關(guān)注小編的公眾號【Python日志】民逼,或者在后臺私信小編都可以的喲
開發(fā)工具
Python版本:3.7.8
相關(guān)模塊:
DecryptLogin模塊泵殴;
argparse模塊;
requests模塊拼苍;
pyttsx3模塊笑诅;
prettytable模塊;
以及一些python自帶的模塊疮鲫。
環(huán)境搭建
安裝Python并添加到環(huán)境變量吆你,pip安裝需要的相關(guān)模塊即可。
原理簡介
首先俊犯,要搶購淘寶商品妇多,肯定要先模擬登錄淘寶,這里我們還是借助公眾號之前開源的DecryptLogin包:
項目地址:https://github.com/CharlesPikachu/DecryptLogin
安裝:pip install DecryptLogin
具體而言燕侠,代碼實現(xiàn)如下:
from DecryptLogin import login
'''模擬登錄'''
def login(self):
lg = login.Login()
infos_return, session = lg.taobao()
return infos_return, session
接著者祖,我們來考慮一下如何獲得想要搶購的商品的信息,是不是可以直接添加到購物車绢彤,然后通過查詢購物車信息來獲得我們想要搶購的商品信息呢七问?寫個代碼簡單測試一下:
'''獲得購物車信息'''
def buycartinfo(self):
url = 'https://cart.taobao.com/cart.htm'
headers = {
'user-agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/87.0.4280.88 Safari/537.36',
'sec-fetch-dest': 'document', 'sec-fetch-mode': 'navigate', 'sec-fetch-site': 'none', 'sec-fetch-user': '?1',
'upgrade-insecure-requests': '1',
'accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9',
'accept-encoding': 'gzip, deflate, br', 'accept-language': 'zh-CN,zh;q=0.9,en;q=0.8',
'cache-control': 'max-age=0'
}
response = self.session.get(url, headers=headers)
print(response.text)
發(fā)現(xiàn)返回的數(shù)據(jù)里是有我們需要的信息的:
因為請求這個接口的時候我的淘寶購物車是這樣子的:
剛好是對應(yīng)上的,說明我們抓到的接口沒毛病茫舶,只需要進一步把我們需要的信息提取出來即可:
# 解析購物車信息
if len(cart_infos['list']) == 0:
raise RuntimeError('購物車是空的, 請在購物車中添加需要搶購的商品')
good_infos = {}
for idx, item in enumerate(cart_infos['list']):
good_info = {
'title': item['bundles'][0]['orders'][0]['title'],
'cart_id': item['bundles'][0]['orders'][0]['cartId'],
'cart_params': item['bundles'][0]['orders'][0]['cartActiveInfo']['cartBcParams'],
'item_id': item['bundles'][0]['orders'][0]['itemId'],
'sku_id': item['bundles'][0]['orders'][0]['skuId'],
'seller_id': item['bundles'][0]['orders'][0]['sellerId'],
'to_buy_info': item['bundles'][0]['orders'][0]['toBuyInfo'],
}
good_infos[str(idx)] = good_info
# 打印并選擇想要搶購的商品信息
title, items = ['id', 'title'], []
for key, value in good_infos.items():
items.append([key, value['title']])
self.printTable(title, items)
good_id = input('請選擇想要搶購的商品編號(例如"0"): ')
assert good_id in good_infos, '輸入的商品編號有誤'
效果大概是這樣的:用戶根據(jù)抓取到的自己購物車中的已有的商品信息來選擇自己想要搶購的商品(即械巡,如果你想要搶購某件商品,你需要先把它加入自己的購物車中)饶氏。接下來讥耗,我們需要考慮的就是如何搶購選中的商品了。具體而言嚷往,購買商品的代碼實現(xiàn)如下:
'''購買商品'''
def buygood(self, info, user_id):
# 發(fā)送結(jié)算請求
url = 'https://buy.taobao.com/auction/order/confirm_order.htm?spm=a1z0d.6639537.0.0.undefined'
headers = {
'cache-control': 'max-age=0', 'upgrade-insecure-requests': '1',
'user-agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/87.0.4280.88 Safari/537.36',
'origin': 'https://cart.taobao.com', 'content-type': 'application/x-www-form-urlencoded',
'accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9',
'sec-fetch-site': 'same-site', 'sec-fetch-mode': 'navigate', 'sec-fetch-user': '?1',
'sec-fetch-dest': 'document', 'referer': 'https://cart.taobao.com/',
'accept-encoding': 'gzip, deflate, br', 'accept-language': 'zh-CN,zh;q=0.9,en;q=0.8'
}
cart_id, item_id, sku_id, seller_id, cart_params, to_buy_info = info['cart_id'], info['item_id'], info['sku_id'], info['seller_id'], info['cart_params'], info['to_buy_info']
data = {
'item': f'{cart_id}_{item_id}_1_{sku_id}_{seller_id}_0_0_0_{cart_params}_{urllib.parse.quote(str(to_buy_info))}__0',
'buyer_from': 'cart',
'source_time': ''.join(str(int(time.time() * 1000)))
}
response = self.session.post(url = url, data = data, headers = headers, verify = False)
order_info = re.search('orderData= (.*?);\n</script>', response.text).group(1)
order_info = json.loads(order_info)
# 發(fā)送提交訂單請求
token = self.session.cookies['_tb_token_']
endpoint = order_info['endpoint']
data = order_info['data']
structure = order_info['hierarchy']['structure']
hierarchy = order_info['hierarchy']
linkage = order_info['linkage']
linkage.pop('url')
submitref = order_info['data']['submitOrderPC_1']['hidden']['extensionMap']['secretValue']
sparam1 = order_info['data']['submitOrderPC_1']['hidden']['extensionMap']['sparam1']
input_charset = order_info['data']['submitOrderPC_1']['hidden']['extensionMap']['input_charset']
event_submit_do_confirm = order_info['data']['submitOrderPC_1']['hidden']['extensionMap']['event_submit_do_confirm']
url = f'https://buy.taobao.com/auction/confirm_order.htm?x-itemid={item_id}&x-uid={user_id}&submitref={submitref}&sparam1={sparam1}'
data_submit = {}
for key, value in data.items():
if value.get('submit') == 'true' or value.get('submit'):
data_submit[key] = value
data = {
'action': '/order/multiTerminalSubmitOrderAction',
'_tb_token_': token,
'event_submit_do_confirm': '1',
'praper_alipay_cashier_domain': 'cashierrz54',
'input_charset': 'utf-8',
'endpoint': urllib.parse.quote(json.dumps(endpoint)),
'data': urllib.parse.quote(json.dumps(data_submit)),
'hierarchy': urllib.parse.quote(json.dumps({"structure": structure})),
'linkage': urllib.parse.quote(json.dumps(linkage))
}
headers = {
'cache-control': 'max-age=0', 'upgrade-insecure-requests': '1', 'origin': 'https://buy.taobao.com',
'content-type': 'application/x-www-form-urlencoded',
'user-agent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/87.0.4280.88 Safari/537.36',
'accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9',
'sec-fetch-site': 'same-origin', 'sec-fetch-mode': 'navigate', 'sec-fetch-user': '?1',
'sec-fetch-dest': 'document',
'referer': 'https://buy.taobao.com/auction/order/confirm_order.htm?spm=a1z0d.6639537.0.0.undefined',
'accept-encoding': 'gzip, deflate, br', 'accept-language': 'zh-CN,zh;q=0.9,en;q=0.8'
}
response = self.session.post(url, data=data, headers=headers)
if response.status_code == 200: return True
return False
搶購的邏輯如下葛账,即每隔一段時間嘗試購買一次商品,直到成功為止:
# 根據(jù)選擇嘗試購買商品
print(f'[INFO]: 正在嘗試搶購商品{good_infos[good_id]["title"]}')
while True:
try:
is_success = self.buygood(good_infos[good_id], user_id)
except:
is_success = False
if is_success: break
time.sleep(self.trybuy_interval)
print(f'[INFO]: 搶購{good_infos[good_id]["title"]}成功, 已為您自動提交訂單, 請盡快前往淘寶完成付款')
最終淘寶里的效果如下:
運行的終端的效果類似這樣:
當(dāng)然這樣提示的效果可能不明顯皮仁,畢竟你不會一直盯著這個終端看籍琳,你可以加個語音提示菲宴,代碼如下:
# 電腦語音提示
for _ in range(5):
pyttsx3.speak('已經(jīng)為您搶購到你所需的商品, 請盡快前往淘寶完成付款.')
當(dāng)然,也許你也不一定一直在電腦邊上趋急,所以你還可以加個server醬提示喝峦,可以在商品搶購成功之后將該消息發(fā)送到你的微信上,代碼如下:
'''發(fā)送Server醬提示'''
def pushwechat(self, desp='已經(jīng)為您搶購到你所需的商品, 請盡快前往淘寶完成付款.'):
server_url = f'https://sc.ftqq.com/{self.server_key}.send'
params = {
'text': '商品搶購成功提示',
'desp': desp,
}
response = requests.get(server_url, params=params)
return response
server醬配置地址如下:
ok呜达,大功告成啦谣蠢,完整源代碼詳見相關(guān)文件。
效果展示
使用方式:
usage: taobaosnap.py [-h] --interval INTERVAL [--key KEY]
淘寶搶購腳本
optional arguments:
-h, --help show this help message and exit
--interval INTERVAL 搶購商品時查詢商品是否可以購買的時間間隔(單位秒)
--key KEY Server醬的Key
效果如下:
server醬的通知效果如下:
OK查近,今天的代碼就到這里結(jié)束了眉踱,要是有小伙伴看不懂的話,或者需要源碼的都可以私信小編的哈霜威,或者關(guān)注小編的公眾號【Python日志】谈喳,可以了領(lǐng)取的哈
大家記得給小編點贊+關(guān)注喲