618!python京东 PC 版抢券程序

前言

本文的文字及图片来源于网络,仅供学习、交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理。

作者:python技术

PS:如有需要Python学习资料的小伙伴可以加点击下方链接自行获取

python免费学习资料以及群交流解答点击即可加入


618!京东 PC 版抢卷

618不到一个月了,京东 618 活动已经开始了,不知道小伙伴们参与了没。先抢一下京东的优惠劵,也许 618 出现神劵呢?

抓包分析登录

获取二维码图片

这里我们使用二维码登录,通过浏览器的控制台可以看出有个带 show 的 url 地址可以得到二维码登录图片
在这里插入图片描述
首先导入所有需要的包和一个解析 json 的方法

import requests
import random
import time
import os
import json

from PIL import Image

def parse_json(str):
    return json.loads(str[str.find('{'):str.rfind('}') + 1])

显示京东登录二维码图片

user_agent = 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_4) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/81.0.4044.138 Safari/537.36'

session = requests.session()

def show_QRcode():
    url = 'https://qr.m.jd.com/show'
    params = {
        'appid': 133,
        'size': 147,
        't': str(int(time.time() * 1000)),
    }
    headers = {
        'User-Agent': user_agent,
        'Referer': 'https://passport.jd.com/new/login.aspx',
    }
    resp = session.get(url=url, headers=headers, params=params)

    QRcode_path = 'QRcode.png'
    with open(QRcode_path, 'wb') as f:
        for chunk in resp.iter_content(chunk_size=1024):
            f.write(chunk)

    QRcode = Image.open(QRcode_path)
    QRcode.show()

检测二维码是否扫码

获取二维码图片之后,在 Fiddler 抓包神器中一直刷新带 check 的 url 地址,猜测这个也是和登录有关的,应该是检测二维码是否被扫码。
在这里插入图片描述
检测是否被扫码抓包

def check_QRcode():
    
    url = 'https://qr.m.jd.com/check'
    params = {
        'appid': '133',
        'callback': 'jQuery{}'.format(random.randint(1000000, 9999999)),
        'token': session.cookies.get('wlfstk_smdl'),
        '_': str(int(time.time() * 1000)),
    }
    headers = {
        'User-Agent': user_agent,
        'Referer': 'https://passport.jd.com/new/login.aspx?ReturnUrl=https%3A%2F%2Fwww.jd.com%2F',
    }
    resp = session.get(url=url, headers=headers, params=params)
    resp_json = parse_json(resp.text)

    if 'ticket' in resp_json:
        print(resp_json)
        return resp_json['ticket']
    else:
        print(resp_json['msg'])
        print('请刷新京东登录二维码!')
        os._exit(0)

示例结果

{'code': 200, 'ticket': 'AAEAIPL-bkU4RNrUw7YaLqYZhjWKqvP23PtkY1XTD2Cv2a52'}

验证二维码

在检测二维码是否被扫码之后,在 Fiddler 抓包神器中出现有一个带 qrCodeTicketValidation 的 url 地址,需要传递一个 t 参数,这个参数正好是 https://qr.m.jd.com/check 的返回值
在这里插入图片描述
验证二维码抓包

def validation_QRcode(ticket):
    url = 'https://passport.jd.com/uc/qrCodeTicketValidation'
    headers = {
        'User-Agent': user_agent,
        'Referer': 'https://passport.jd.com/new/login.aspx?ReturnUrl=https%3A%2F%2Fwww.jd.com%2F',
    }
    params={
        't': ticket
    }
    resp = session.get(url=url, headers=headers, params=params)
    print(resp.text)

示例结果

{"returnCode":0,"url":"https://www.jd.com/"}

到这里京东二维码已经登录成功了

领取优惠券

在这里插入图片描述
可以看到这个 url 的参数有 page,pageSize,不用猜都知道这是刷新的优惠券列表,抓它

def coupon_list():
    url = 'https://a.jd.com/indexAjax/getCouponListByCatalogId.html'
    headers = {
        'User-Agent': user_agent,
        'Referer': 'https://a.jd.com/?cateId=118',
    }
    couponList = []
    for i in range(1, 20):
        params = {
            'callback': 'jQuery{}'.format(random.randint(1000000, 9999999)),
            'catalogId': '118',
            'page': str(i),
            'pageSize': '9',
            '_': str(int(time.time() * 1000)),
        }
        try:
            resp = session.get(url=url, params=params, headers=headers)
            json = parse_json(resp.text)
            couponList.extend(json['couponList'])
            if json['totalNum'] == 1:
                continue
            else:
                break
        except Exception:
            print('出错了!')
    return couponList

示例结果就不写了,返回的 json 串太长了

最后一步领取优惠劵,在浏览器控制台中 url 地址上有一个 key,这个 key 存在与优惠劵列表的返回值中
在这里插入图片描述
最后一步,领取优惠券

def get_coupon(coupon_list):
    url = 'https://a.jd.com/indexAjax/getCoupon.html'
    headers = {
        'User-Agent': user_agent,
        'Referer': 'https://a.jd.com/?cateId=118',
    }
    for coupon in coupon_list:
        params = {
            'callback': 'jQuery{}'.format(random.randint(1000000, 9999999)),
            'key': coupon['key'],
            'type': '1',
            '_': str(int(time.time() * 1000)),
        }
        time.sleep(1)
        resp = session.get(url=url, params=params, headers=headers)
        print(resp.text)

示例结果

jQuery1912666({"code":"15","success":false,"message":"您今天已经参加过此活动,别太贪心哟,明天再来~"})
jQuery3381540({"code":"15","success":false,"message":"您今天已经参加过此活动,别太贪心哟,明天再来~"})
jQuery6247320({"code":"16","success":false,"message":"本时段优惠券已抢完,请14:00再来吧!"})
jQuery5888701({"code":"15","success":false,"message":"您今天已经参加过此活动,别太贪心哟,明天再来~"})
jQuery5048959({"code":"14","success":false,"message":"您已经参加过此活动,别太贪心哟,下次再来~"})
jQuery8608381({"code":"14","success":false,"message":"您已经参加过此活动,别太贪心哟,下次再来~"})
jQuery2539212({"code":"15","success":false,"message":"您今天已经参加过此活动,别太贪心哟,明天再来~"})
jQuery6439595({"code":"999","success":true,"message":"领券成功"})
jQuery2972325({"code":"15","success":false,"message":"您今天已经参加过此活动,别太贪心哟,明天再来~"})
jQuery1697862({"code":"15","success":false,"message":"您今天已经参加过此活动,别太贪心哟,明天再来~"})
jQuery1905738({"code":"15","success":false,"message":"您今天已经参加过此活动,别太贪心哟,明天再来~"})
jQuery7639432({"code":"15","success":false,"message":"您今天已经参加过此活动,别太贪心哟,明天再来~"})
jQuery8239706({"code":"15","success":false,"message":"您今天已经参加过此活动,别太贪心哟,明天再来~"})
jQuery1221489({"code":"15","success":false,"message":"您今天已经参加过此活动,别太贪心哟,明天再来~"})

完整代码

import requests
import random
import time
import os
import json

from PIL import Image

user_agent = 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_4) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/81.0.4044.138 Safari/537.36'

session = requests.session()


def show_QRcode():
    url = 'https://qr.m.jd.com/show'
    params = {
        'appid': 133,
        'size': 147,
        't': str(int(time.time() * 1000)),
    }
    headers = {
        'User-Agent': user_agent,
        'Referer': 'https://passport.jd.com/new/login.aspx',
    }
    resp = session.get(url=url, headers=headers, params=params)

    QRcode_path = 'QRcode.png'
    with open(QRcode_path, 'wb') as f:
        for chunk in resp.iter_content(chunk_size=1024):
            f.write(chunk)

    QRcode = Image.open(QRcode_path)
    QRcode.show()

def check_QRcode():
    
    url = 'https://qr.m.jd.com/check'
    params = {
        'appid': '133',
        'callback': 'jQuery{}'.format(random.randint(1000000, 9999999)),
        'token': session.cookies.get('wlfstk_smdl'),
        '_': str(int(time.time() * 1000)),
    }
    headers = {
        'User-Agent': user_agent,
        'Referer': 'https://passport.jd.com/new/login.aspx?ReturnUrl=https%3A%2F%2Fwww.jd.com%2F',
    }
    resp = session.get(url=url, headers=headers, params=params)
    resp_json = parse_json(resp.text)

    if 'ticket' in resp_json:
        print(resp_json)
        return resp_json['ticket']
    else:
        print(resp_json['msg'])
        print('请刷新JD登录二维码!')
        os._exit(0)


def validation_QRcode(ticket):

    url = 'https://passport.jd.com/uc/qrCodeTicketValidation'
    headers = {
        'User-Agent': user_agent,
        'Referer': 'https://passport.jd.com/new/login.aspx?ReturnUrl=https%3A%2F%2Fwww.jd.com%2F',
    }
    params={
        't': ticket
    }
    resp = session.get(url=url, headers=headers, params=params)
    print(resp.text)


def parse_json(str):
    return json.loads(str[str.find('{'):str.rfind('}') + 1])


def coupon_list():
    url = 'https://a.jd.com/indexAjax/getCouponListByCatalogId.html'
    headers = {
        'User-Agent': user_agent,
        'Referer': 'https://a.jd.com/?cateId=118',
    }
    couponList = []
    for i in range(1, 20):
        params = {
            'callback': 'jQuery{}'.format(random.randint(1000000, 9999999)),
            'catalogId': '118',
            'page': str(i),
            'pageSize': '9',
            '_': str(int(time.time() * 1000)),
        }
        try:
            resp = session.get(url=url, params=params, headers=headers)
            json = parse_json(resp.text)
            couponList.extend(json['couponList'])
            if json['totalNum'] == 1:
                continue
            else:
                break
        except Exception:
            print('出错了!')
    return couponList


def get_coupon(coupon_list):
    url = 'https://a.jd.com/indexAjax/getCoupon.html'
    headers = {
        'User-Agent': user_agent,
        'Referer': 'https://a.jd.com/?cateId=118',
    }
    for coupon in coupon_list:
        params = {
            'callback': 'jQuery{}'.format(random.randint(1000000, 9999999)),
            'key': coupon['key'],
            'type': '1',
            '_': str(int(time.time() * 1000)),
        }
        time.sleep(1)
        resp = session.get(url=url, params=params, headers=headers)
        print(resp.text)



if __name__ == '__main__':
    show_QRcode()

    time.sleep(10)

    ticket = check_QRcode()
    validation_QRcode(ticket)
    coupon_list = coupon_list()
    get_coupon(coupon_list)

总结

京东PC版抓取优惠券到这里就结束了,代码还有不完善的地方,比如每次都要扫码登录,没有保存 cookies 自动登录京东等待。小伙伴们可以根据自身需求更改代码,造出适合自己的轮子。

  • 7
    点赞
  • 50
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
# JD_AutoBuy ## 京东抢购 Python爬虫,自动登录京东网站,查询商品库存,价格,显示购物车详情等。 可以指定抢购商品,自动购买下单,然后手动去京东付款就行。 ## chang log + 2017-03-30 实现二维码扫码登陆 ## 运行环境 Python 2.7 ## 第三方库 - [Requests][1]: 简单好用,功能强大的Http请求库 - [beautifulsoup4][2]: HTML文档格式化及便签选择器 ## 环境配置 ``` Python pip install requests pip install beautifulsoup4 ``` ## 使用帮助 ``` cmd > python scraper-jd.py -h usage: scraper-jd.py [-h] [-u USERNAME] [-p PASSWORD] [-g GOOD] [-c COUNT] [-w WAIT] [-f] [-s] Simulate to login Jing Dong, and buy sepecified good optional arguments: -h, --help show this help message and exit -u USERNAME, --username USERNAME Jing Dong login user name -p PASSWORD, --password PASSWORD Jing Dong login user password -g GOOD, --good GOOD Jing Dong good ID -c COUNT, --count COUNT The count to buy -w WAIT, --wait WAIT Flush time interval, unit MS -f, --flush Continue flash if good out of stock -s, --submit Submit the order to Jing Dong ``` ## 实例输出 ``` cmd +++++++++++++++++++++++++++++++++++++++++++++++++++++++ Thu Mar 30 17:10:01 2017 > 请打开京东手机客户端,准备扫码登陆: 201 : 二维码未扫描 ,请扫描二维码 201 : 二维码未扫描 ,请扫描二维码 201 : 二维码未扫描 ,请扫描二维码 201 : 二维码未扫描 ,请扫描二维码 202 : 请手机客户端确认登录 200 : BADACIFYhf6fakfHvjiYTlwGzSp4EjFATN3Xw1ePR1hITtw0 登陆成功 +++++++++++++++++++++++++++++++++++++++++++++++++++++++ Thu Mar 30 17:10:28 2017 > 商品详情 编号:3133857 库存:现货 价格:6399.00 名称:Apple iPhone 7 Plus (A1661) 128G 黑色 移动联通电信4G手机 链接:http://cart.jd.com/gate.action?pid=3133857&pcount=1&ptype=1 商品已成功加入购物车! 购买数量:3133857 > 1 +++++++++++++++++++++++++++++++++++++++++++++++++++++++ Thu Mar 30 17:10:30 2017 > 购物车明细 购买 数量 价格 总价 商品 Y 1 6399.00 6399.00 Apple iPhone 7 Plus (A1661) 128G 黑色 移动联通电信4G手机 总数: 1 总额: 6399.00 +++++++++++++++++++++++++++++++++++++++++++++++++++++++ Thu Mar 30 17:10:30 2017 > 订单详情 +++++++++++++++++++++++++++++++++++++++++++++++++++++++ ... ``` ## 注 代码仅供学习之用,京东网页不断变化,代

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值