經過無數次搶購失敗後,發現商家會不定時的放出少量貨源,目測每次會有幾台。如果我們編寫一個腳本程式24小時不間斷監聽商品庫存,一旦查詢到貨源便開始嘗試自動下單,這樣就可以大大提高我們的成功機率。
京東對於商品的搶購主要分為兩種:
#預約搶購:到點開放購買,和普通商品下單流程一致;秒殺商品:單獨的搶購介面和下單流程。
當然本次針對的預約搶購類或無貨訂購類,即整體下單流程和購買普通商品時一樣:
登錄帳號→ 進入購物車→選擇搶購商品→ 點擊去結算→ 點擊提交訂單→ 選擇付款方式並付款。
由於筆者本身沒有一個可以抓包的客戶端,決定採用京東 WEB 端介面實作我們的腳本程式。
於是經過京東網頁下單一流程的分析,將我們的腳本程式分為四個模組:帳號登入模組、庫存監聽模組、購物車管理模組、訂單管理模組。
由於使用帳號密碼時有驗證碼限制,此處採用掃碼登入方式繞過。
如對掃碼登入不熟悉或有興趣的同學可以查看周週之前的博文 掃碼登入原則和實作。
這次只要針對京東登入頁進行抓包分析,找到幾個有用介面:
def getQRcode(self):
url = 'https://qr.m.jd.com/show'
payload = {
'appid': 133,
'size': 147,
't': str(int(time.time() * 1000)),
}
headers = {
'User-Agent': self.userAgent,
'Referer': 'https://passport.jd.com/new/login.aspx',
}
resp = self.sess.get(url=url, headers=headers, params=payload)
if not self.respStatus(resp):
return None
return resp.contentdef getQRcodeTicket(self):
url = 'https://qr.m.jd.com/check'
payload = {
'appid': '133',
'callback': 'jQuery{}'.format(random.randint(1000000, 9999999)),
'token': self.sess.cookies.get('wlfstk_smdl'),
'_': str(int(time.time() * 1000)),
}
headers = {
'User-Agent': self.userAgent,
'Referer': 'https://passport.jd.com/new/login.aspx',
}
resp = self.sess.get(url=url, headers=headers, params=payload)
if not self.respStatus(resp):
return False
respJson = self.parseJson(resp.text)
if respJson['code'] != 200:
return None
else:
return respJson['ticket']def getQRcodeTicket(self):
url = 'https://qr.m.jd.com/check'
payload = {
'appid': '133',
'callback': 'jQuery{}'.format(random.randint(1000000, 9999999)),
'token': self.sess.cookies.get('wlfstk_smdl'),
'_': str(int(time.time() * 1000)),
}
headers = {
'User-Agent': self.userAgent,
'Referer': 'https://passport.jd.com/new/login.aspx',
}
resp = self.sess.get(url=url, headers=headers, params=payload)
if not self.respStatus(resp):
return False
respJson = self.parseJson(resp.text)
if respJson['code'] != 200:
return None
else:
return respJson['ticket']此時驗證Ticket 有效後使用pickle 庫將程式會話中的cookie 儲存到本地以便下次使用。
庫存監聽較為簡單,分析商品詳情頁,取得店舖ID以及商品分類屬性:
def getItemDetail(self, skuId):
url = 'https://item.jd.com/{}.html'.format(skuId)
page = requests.get(url=url, headers=self.headers)
html = etree.HTML(page.text)
vender = html.xpath(
'//div[@class="follow J-follow-shop"]/@data-vid')[0]
cat = html.xpath('//a[@clstag="shangpin|keycount|product|mbNav-3"]/@href')[
0].replace('//list.jd.com/list.html?cat=', '')
if not vender or not cat:
raise Exception('获取商品信息失败,请检查SKU是否正确')
detail = dict(catId=cat, venderId=vender)
return detaildef getItemStock(self, skuId, num, areaId):
item = self.itemDetails.get(skuId)
if not item:
return False
url = 'https://c0.3.cn/stock'
payload = {
'skuId': skuId,
'buyNum': num,
'area': areaId,
'ch': 1,
'_': str(int(time.time() * 1000)),
'callback': 'jQuery{}'.format(random.randint(1000000, 9999999)),
# get error stock state without this param
'extraParam': '{"originid":"1"}',
# get 403 Forbidden without this param (obtained from the detail page)
'cat': item.get('catId'),
# return seller information with this param (can't be ignored)
'venderId': item.get('venderId')
}
headers = {
'User-Agent': self.userAgent,
'Referer': 'https://item.jd.com/{}.html'.format(skuId),
}
respText = ''
try:
respText = requests.get(
url=url, params=payload, headers=headers, timeout=self.timeout).text
respJson = self.parseJson(respText)
stockInfo = respJson.get('stock')
skuState = stockInfo.get('skuState') # 商品是否上架
# 商品库存状态:33 -- 现货 0,34 -- 无货 36 -- 采购中 40 -- 可配货
stockState = stockInfo.get('StockState')
return skuState == 1 and stockState in (33, 40)無貨商品加入購物車我們是無法透過頁面操作的,我們這邊可以使用其他有貨商品進行嘗試,主要查看購物車的增刪改查介面:
def uncheckCartAll(self):
""" 取消所有选中商品
return 购物车信息
"""
url = 'https://api.m.jd.com/api'
headers = {
'User-Agent': self.userAgent,
'Content-Type': 'application/x-www-form-urlencoded',
'origin': 'https://cart.jd.com',
'referer': 'https://cart.jd.com'
}
data = {
'functionId': 'pcCart_jc_cartUnCheckAll',
'appid': 'JDC_mall_cart',
'body': '{"serInfo":{"area":"","user-key":""}}',
'loginType': 3
}
resp = self.sess.post(url=url, headers=headers, data=data)
# return self.respStatus(resp) and resp.json()['success']
return respdef addCartSku(self, skuId, skuNum):
""" 加入购入车
skuId 商品sku
skuNum 购买数量
retrun 是否成功
"""
url = 'https://api.m.jd.com/api'
headers = {
'User-Agent': self.userAgent,
'Content-Type': 'application/x-www-form-urlencoded',
'origin': 'https://cart.jd.com',
'referer': 'https://cart.jd.com'
}
data = {
'functionId': 'pcCart_jc_cartAdd',
'appid': 'JDC_mall_cart',
'body': '{\"operations\":[{\"carttype\":1,\"TheSkus\":[{\"Id\":\"' + skuId + '\",\"num\":' + str(skuNum) + '}]}]}',
'loginType': 3
}
resp = self.sess.post(url=url, headers=headers, data=data)
return self.respStatus(resp) and resp.json()['success']def changeCartSkuCount(self, skuId, skuUid, skuNum, areaId):
""" 修改购物车商品数量
skuId 商品sku
skuUid 商品用户关系
skuNum 购买数量
retrun 是否成功
"""
url = 'https://api.m.jd.com/api'
headers = {
'User-Agent': self.userAgent,
'Content-Type': 'application/x-www-form-urlencoded',
'origin': 'https://cart.jd.com',
'referer': 'https://cart.jd.com'
}
body = '{\"operations\":[{\"TheSkus\":[{\"Id\":\"'+skuId+'\",\"num\":'+str(
skuNum)+',\"skuUuid\":\"'+skuUid+'\",\"useUuid\":false}]}],\"serInfo\":{\"area\":\"'+areaId+'\"}}'
data = {
'functionId': 'pcCart_jc_changeSkuNum',
'appid': 'JDC_mall_cart',
'body': body,
'loginType': 3
}
resp = self.sess.post(url=url, headers=headers, data=data)
return self.respStatus(resp) and resp.json()['success']以上是我們一次購買需要用到的最少接口,為了不破壞帳戶購物車中已有數據,採用一下步驟準備好購物車:
取消全部勾選(返回購物車資訊);已在購物車則修改商品數量;不在購物車則加入購物車。 3.4 訂單操作
當我們準備好購物車之後(選取購買商品以及調整購買數量),就可以進行下一步訂單相關操作:
def getCheckoutPage(self):
"""获取订单结算页面信息
:return: 结算信息 dict
"""
url = 'http://trade.jd.com/shopping/order/getOrderInfo.action'
# url = 'https://cart.jd.com/gotoOrder.action'
payload = {
'rid': str(int(time.time() * 1000)),
}
headers = {
'User-Agent': self.userAgent,
'Referer': 'https://cart.jd.com/cart',
}def submitOrder(self):
"""提交订单
:return: True/False 订单提交结果
"""
url = 'https://trade.jd.com/shopping/order/submitOrder.action'
# js function of submit order is included in https://trade.jd.com/shopping/misc/js/order.js?r=2018070403091
data = {
'overseaPurchaseCookies': '',
'vendorRemarks': '[]',
'submitOrderParam.sopNotPutInvoice': 'false',
'submitOrderParam.trackID': 'TestTrackId',
'submitOrderParam.ignorePriceChange': '0',
'submitOrderParam.btSupport': '0',
'riskControl': self.risk_control,
'submitOrderParam.isBestCoupon': 1,
'submitOrderParam.jxj': 1,
'submitOrderParam.trackId': self.track_id,
'submitOrderParam.eid': self.eid,
'submitOrderParam.fp': self.fp,
'submitOrderParam.needCheck': 1,
}以上是如何使用Python編寫自動搶購京東下單腳本?的詳細內容。更多資訊請關注PHP中文網其他相關文章!