使用Python开发个京东上抢口罩的小实例(仅作技术研究学习使用)


Posted in Python onMarch 10, 2020

全国抗”疫”这么久终于见到曙光,在家待了将近一个月,现在终于可以去上班了,可是却发现出门必备的口罩却一直买不到。最近看到京东上每天都会有口罩的秒杀活动,试了几次却怎么也抢不到,到了抢购的时间,浏览器的页面根本就刷新不出来,等刷出来秒杀也结束了。现在每天只放出一万个,却有几百万人在抢,很想知道别人是怎么抢到的,于是就在网上找了大神公开出来的抢购代码。看了下代码并不复杂,现在我们就报着学习的态度一起看看。

使用模块

requests:类似 urllib,主要用于向网站发送 HTTP 请求。

beautifulsoup4:HTML 解析器,用于将 HTML 文档转换成一个复杂的树形结构。

pillow:Python 图像处理标准库,用于识别验证码。

配置文件

一般项目中我们都需要把一些可配置的内容放到配置文件中,现在我们来看下这里主要配置项:

# 邮寄地所属地区ID

area = 123456

# 这是配置的商品的ID

skuid = 6828101

# 打码服务器的地址

captchaUrl = http://xxx/pic

# 通知邮箱

mail = xxxxxx@qq.com

# cookie的设置

cookies_String = shshshfpa21jsda8923892949204923123

OK,有了配置文件,那我们就得有一段读取配置文件的代码,这段代码实现将配置内容加载到内存中。

import os
import configparser
# 加载配置文件
class Config(object):
  def __init__(self, config_file='configDemo.ini'):
    self._path = os.path.join(os.getcwd(), config_file)
    if not os.path.exists(self._path):
      raise FileNotFoundError("No such file: config.ini")
    self._config = configparser.ConfigParser()
    self._config.read(self._path, encoding='utf-8-sig')
    self._configRaw = configparser.RawConfigParser()
    self._configRaw.read(self._path, encoding='utf-8-sig')
  def get(self, section, name):
    return self._config.get(section, name)
  def getRaw(self, section, name):
    return self._configRaw.get(section, name)

主程序模块

我看 GitHub 上也有实现了运行程序后通过京东 App 扫码登陆,然后再通过登陆 Cookie 访问网站的,不过这里并没有使用这种方式,毕竟我们打开浏览器开发者工具也能很容易获取到登陆的 Cookie,这里就是将 Cookie 直接放到配置文件里的方式。

# 主程序入口
# 检查是否存在要抢购的端口,然后进入循环扫描
if len(skuids) != 1:
  logger.info('请准备一件商品')
skuId = skuids[0]
flag = 1
# 循环扫描该商品是否有货,有库存即会自动下单,无库存则休眠后继续扫描
while (1):
  try:
    # 初始化校验
    if flag == 1:
      logger.info('当前是V3版本')
      validate_cookies()  # 校验登陆状态
      getUsername()    # 获取登陆用户信息
      select_all_cart_item()  # 全选购物车
      remove_item()      # 删除购物车
      add_item_to_cart(skuId)  # 增加抢购的商品
    # 检测配置文件修改
    if int(time.time()) - configTime >= 60:
      check_Config()
    logger.info('第' + str(flag) + '次 ')
    # 计数器
    flag += 1
    # 检查库存模块
    inStockSkuid = check_stock(checksession, skuids, area)
    # 自动下单模块
    V3AutoBuy(inStockSkuid)
    # 休眠模块
    timesleep = random.randint(1, 3) / 10
    time.sleep(timesleep)
    # 校验是否还在登录模块
    if flag % 100 == 0:
      V3check(skuId)
  except Exception as e:
    print(traceback.format_exc())
    time.sleep(10)

以上就是该项目主程序,我已经将代码在原来基础上增加了些注释,可以让我们更容易明白代码的含义。下面我们就选择几个比较关键的代码分析一下。

# 校验登陆状态
def validate_cookies():
  for flag in range(1, 3):
    try:
      targetURL = 'https://order.jd.com/center/list.action'
      payload = {
        'rid': str(int(time.time() * 1000)),
      }
      resp = session.get(url=targetURL, params=payload, allow_redirects=False)
      if resp.status_code == requests.codes.OK:
        logger.info('登录成功')
        return True
      else:
        logger.info('第【%s】次请重新获取cookie', flag)
        time.sleep(5)
        continue
    except Exception as e:
      logger.info('第【%s】次请重新获取cookie', flag)
      time.sleep(5)
      continue
  message.sendAny('脚本登录cookie失效了,请重新登录')
  sys.exit(1)

以上代码是每次调用时,循环两次获取通过 session 获取当前登陆状态,如果两次后依然失败则退出程序。

添加商品到购物车

接下来我们再看下如果添加商品到购物车的,代码如下:

def add_item_to_cart(sku_id):
  # 请求添加商品url
  url = 'https://cart.jd.com/gate.action'
  payload = {
    'pid': sku_id,
    'pcount': 1,
    'ptype': 1,
  }
  # 返回结果
  resp = session.get(url=url, params=payload)
  # 套装商品加入购物车后直接跳转到购物车页面
  if 'https://cart.jd.com/cart.action' in resp.url:
    result = True
  else:
   # 普通商品成功加入购物车后会跳转到提示 "商品已成功加入购物车!" 页面
    soup = BeautifulSoup(resp.text, "html.parser")
    result = bool(soup.select('h3.ftx-02')) # [<h3 class="ftx-02">商品已成功加入购物车!</h3>]
  if result:
    logger.info('%s 已成功加入购物车', sku_id)
  else:
    logger.error('%s 添加到购物车失败', sku_id)

在这里,只是简单几行代码就能将端口添加到购物车了,而且这里还区分了不同类型商品添加到购物车返回的页面结果是不同的,所以要进行区别处理。

购买商品

将商品添加到购物车了,接下来我们就得提交结算页了,也就是将商品提交到付款页面,这段代码有点多,我简化了下并加了些注释:

def submit_order(session, risk_control, sku_id, skuids, submit_Time, encryptClientInfo, is_Submit_captcha, payment_pwd,
         submit_captcha_text, submit_captcha_rid):
  # 提交端口的url
  url = 'https://trade.jd.com/shopping/order/submitOrder.action'
  # 提交参数
  data = {
    'overseaPurchaseCookies': '',
    'vendorRemarks': '[]',
    'submitOrderParam.sopNotPutInvoice': 'false',
    'submitOrderParam.trackID': 'TestTrackId',
    'submitOrderParam.ignorePriceChange': '0',
    'submitOrderParam.btSupport': '0',
    'riskControl': risk_control,
    'submitOrderParam.isBestCoupon': 1,
    'submitOrderParam.jxj': 1,
    'submitOrderParam.trackId': '9643cbd55bbbe103eef18a213e069eb0', # Todo: need to get trackId
    'submitOrderParam.needCheck': 1,
  }
  # 如果用到京豆会需要输入支付密码
  def encrypt_payment_pwd(payment_pwd):
    return ''.join(['u3' + x for x in payment_pwd])
  # 校验支付密码
  if len(payment_pwd) > 0:
    data['submitOrderParam.payPassword'] = encrypt_payment_pwd(payment_pwd)
  # 请求报文头
  headers = {
    "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/75.0.3770.100 Safari/531.36",
    "Accept": "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3",
    "Referer": "http://trade.jd.com/shopping/order/getOrderInfo.action",
    "Connection": "keep-alive",
    'Host': 'trade.jd.com',
  }
  # 订单提交会尝试两次
  for count in range(1, 3):
    logger.info('第[%s/%s]次尝试提交订单', count, 3)
    try:
      # 可能会存在的校验码
      if is_Submit_captcha:
        captcha_result = page_detail_captcha(session, encryptClientInfo)
        # 验证码服务错误
        if not captcha_result:
          logger.error('验证码服务异常')
          continue
        data['submitOrderParam.checkcodeTxt'] = submit_captcha_text
        data['submitOrderParam.checkCodeRid'] = submit_captcha_rid
      # 提交订单
      resp = session.post(url=url, data=data, headers=headers)
      resp_json = json.loads(resp.text)
      logger.info('本次提交订单耗时[%s]毫秒', str(int(time.time() * 1000) - submit_Time))
      # 判断是否提交成功
      if resp_json.get('success'):
        logger.info('订单提交成功! 订单号:%s', resp_json.get('orderId'))
        return True
      else:
        # 提交失败返回的多种原因
        resultMessage, result_code = resp_json.get('message'), resp_json.get('resultCode')
        if result_code == 0:
          # self._save_invoice()
          if '验证码不正确' in resultMessage:
            resultMessage = resultMessage + '(验证码错误)'
            logger.info('提交订单验证码[错误]')
            continue
          else:
            resultMessage = resultMessage + '(下单商品可能为第三方商品,将切换为普通发票进行尝试)'
        elif result_code == 60077:
          resultMessage = resultMessage + '(可能是购物车为空 或 未勾选购物车中商品)'
        elif result_code == 60123:
          resultMessage = resultMessage + '(需要在payment_pwd参数配置支付密码)'
        elif result_code == 60070:
          resultMessage = resultMessage + '(省份不支持销售)'
          skuids.remove(sku_id)
          logger.info('[%s]类型口罩不支持销售', sku_id)
        logger.info('订单提交失败, 错误码:%s, 返回信息:%s', result_code, resultMessage)
        logger.info(resp_json)
        return False
    except Exception as e:
      print(traceback.format_exc())
      continue

以上代码实现了商品自动提交到结算页面,这段明显比添加购物车要复杂,果然跟钱有关的都不简单。好了,到了结算页面剩下就是付款了,这个就不需要再抢了,毕竟也没人会抢着给你付钱的。

好了本文主要讲的使用Python开发个京东上抢口罩的小实例只作技术研究学习使用,请不要胡乱使用。更多关于Python模块requests,beautifulsoup4,pillow使用方法请查看下面的相关链接

Python 相关文章推荐
python client使用http post 到server端的代码
Feb 10 Python
windows下安装python paramiko模块的代码
Feb 10 Python
Python接收Gmail新邮件并发送到gtalk的方法
Mar 10 Python
Python Requests模拟登录实现图书馆座位自动预约
Apr 27 Python
在python中实现强制关闭线程的示例
Jan 22 Python
Python面向对象程序设计类的封装与继承用法示例
Apr 12 Python
python pickle存储、读取大数据量列表、字典数据的方法
Jul 07 Python
简单了解python中的与或非运算
Sep 18 Python
python取均匀不重复的随机数方式
Nov 27 Python
python调用HEG工具批量处理MODIS数据的方法及注意事项
Feb 18 Python
Python栈的实现方法示例【列表、单链表】
Feb 22 Python
python 获取当前目录下的文件目录和文件名实例代码详解
Mar 10 Python
python 获取当前目录下的文件目录和文件名实例代码详解
Mar 10 #Python
python爬虫开发之使用Python爬虫库requests多线程抓取猫眼电影TOP100实例
Mar 10 #Python
Django 404、500页面全局配置知识点详解
Mar 10 #Python
python使用gdal对shp读取,新建和更新的实例
Mar 10 #Python
Python实现获取当前目录下文件名代码详解
Mar 10 #Python
python爬虫开发之使用python爬虫库requests,urllib与今日头条搜索功能爬取搜索内容实例
Mar 10 #Python
python+gdal+遥感图像拼接(mosaic)的实例
Mar 10 #Python
You might like
PHP 获取远程文件内容的函数代码
2010/03/24 PHP
php+mysqli事务控制实现银行转账实例
2015/01/29 PHP
PHP弹出对话框技巧详细解读
2015/09/26 PHP
基于PHP的加载类操作以及其他两种魔术方法的应用实例
2017/08/28 PHP
javascript中创建对象的几种方法总结
2013/11/01 Javascript
JavaScript使用push方法添加一个元素到数组末尾用法实例
2015/04/06 Javascript
Bootstrap每天必学之折叠(Collapse)插件
2016/04/25 Javascript
JavaScript中定义对象原型的两种使用方法
2016/12/15 Javascript
js实现百度登录框鼠标拖拽效果
2017/03/07 Javascript
浅谈在koa2中实现页面渲染的全局数据
2017/10/09 Javascript
原生JavaScript实现remove()和recover()功能示例
2018/07/24 Javascript
vue使用localStorage保存登录信息 适用于移动端、PC端
2019/05/27 Javascript
微信小程序获取地理位置及经纬度授权代码实例
2019/09/18 Javascript
Vue组件为什么data必须是一个函数
2020/06/11 Javascript
解决elementui表格操作列自适应列宽
2020/12/28 Javascript
[34:27]DOTA2上海特级锦标赛B组败者赛 VG VS Spirit第一局
2016/02/26 DOTA
python正则匹配抓取豆瓣电影链接和评论代码分享
2013/12/27 Python
Python编程中实现迭代器的一些技巧小结
2016/06/21 Python
python实现汉诺塔方法汇总
2016/07/25 Python
Python 数据可视化pyecharts的使用详解
2019/06/26 Python
Python基于template实现字符串替换
2020/11/27 Python
GIVENCHY纪梵希官方旗舰店:高定彩妆与贵族护肤品
2018/04/16 全球购物
澳大利亚二手奢侈品网站:Modsie
2019/09/23 全球购物
巴西最大的运动品牌:Olympikus
2020/07/14 全球购物
Linux如何命名文件--使用文件名时应注意
2014/05/29 面试题
J2EE中常用的名词进行解释
2015/11/09 面试题
如何写毕业求职自荐信
2013/11/06 职场文书
教师求职推荐信范文
2013/11/20 职场文书
工地质量标语
2014/06/12 职场文书
公司优秀员工获奖感言
2014/08/14 职场文书
群众路线教育实践活动心得体会(教师)
2014/10/31 职场文书
2014收银员工作总结范文
2014/12/16 职场文书
宾馆卫生管理制度
2015/08/06 职场文书
励志语录:你若不勇敢,谁替你坚强
2019/11/08 职场文书
导游词之泰山玉皇顶
2019/12/23 职场文书
PYTHON 使用 Pandas 删除某列指定值所在的行
2022/04/28 Python