lsekfe 发表于 2023-4-25 14:48:33

大麦网演唱会抢票程序之selenium

大麦网上,像林俊杰这类歌星的演唱会的门票,往往时间一到就瞬间售完,网速慢了或者手慢了都只能对着屏幕叹息了。所以有时候你愿意花这钱也不一定买得到看的机会。但是其实这个问题并不难解决,手速慢?那可以让代码模拟操作。网速慢?那可以把程序放到阿里服务器上运行。而这其中的关键就是编写一个抢票程序!
  要说抢票,网上关于12306的抢票程序多如繁星,但是关于大麦网的还真不多,但是核心要义都是一样的:模拟。其实这种程序一定程度上类似于爬虫,所以马上可以想到Python的urllib、BeautifulSoup、selenium、splinter等模块。最初为了方便,偷懒使用了splinter,封装得相当简单,但是功能不多,文档不全,网上资料少,写到后来,感觉主用这个模块解决不了自动登录问题,所以放弃转成了selenium。接下来,我将从账号登录、选择演唱会、购票、确认订单四个方面依次讲解思路。
  1.账号登录
  既然是抢票,我希望给定演唱会信息后一运行程序就能够马上进行,不需要其他的人工干预。这就要求我们的账号能够再抢票前自动登录绑定。其实要实现自动登录,由两种方式:①Cookie ②模拟登录。使用Cookie(原理:如果服务器端的$_COOKIE函数中记录了你的Cookie,那就可以直接调用登录,如果没有就需要人工登录了,登录了之后,二次访问界面就能把自己的Cookie保存到$_COOKIE函数)这种方式其实很方便,保存在本地,要用了调用一下,相当于带了一块出入皇宫的令牌,不用每次进入都验身啥的。但是Cookie涉及过期问题,同时是存在较大安全隐患。所以我决定玩玩另一个方式——模拟登录。
  大麦网的登录界面如下图:
http://www.51testing.com/attachments/2023/04/15326880_202304241321301Pgc1.jpg
  我们需要做的就是填充账号密码,填完了之后点击登录,其实人工登录到这里就完成了,但是用代码填充完内容后点击登录,惊喜就出现了,一个滑块突然出现在按钮上方,并红字提示需要将滑块滑到最右边才能点击登录。所以接下来我们除了要完成内容填充、按钮点击外,还要完成滑块的滑动。(我试过模拟人工填写信息来躲避这个滑块,但是奈何骗不过去,只好直面问题了)
  说实话,因为经验不足,内容填充花了我好几个小时,原因就是这个登录框其实是一个网页,封装在iframe标签中,作为外部网页的子界面,不管我用哪个工具包,用id、class、xpath等哪个方式,都无法定位到账号框和密码框。当我注意到iframe这个坑,才搞明白要用下面这句话来定位到子页面。
  self.driver.switch_to_frame('alibaba-login-box')#里面这个是iframe的id

  接下来就简单了,定位到两个框和一个按钮,然后点击一下按钮,保证滑块乖乖地出现。
  self.driver.find_element_by_id('fm-login-id').send_keys(self.uid)
  self.driver.find_element_by_id('fm-login-password').send_keys(self.upw)
  self.driver.find_element_by_tag_name("button").click()


  然后,想想自己是怎么滑动滑块的,在滑块处按下左键不动,移动鼠标向右,到最右边,然后松手。要模拟这个过程就需要用到ActionChains了。
  ActionChains(self.driver).click_and_hold(self.driver.find_element_by_id('nc_1_n1z')).perform()#按住滑块不动
  ActionChains(self.driver).move_by_offset(xoffset=250, yoffset=0).perform()#直接到终点,可能速度太快,会被系统判错误操作(这也是我不用drag_and_drop这个函数的原因),快到终点时停下
  for i in range(2):
     ActionChains(self.driver).move_by_offset(xoffset=10, yoffset=0).perform()#再慢慢滑两步
     sleep(0.1)
  sleep(0.5)#滑完了之后稍等下,让系统判断完毕
  ActionChains(self.driver).release().perform()#松开点击
  self.driver.find_element_by_tag_name("button").click()#点击登录


  结束了之后,记得加上下面这句话,从iframe切换出去。
  self.driver.switch_to_default_content()

  但是,过了两天,我发现这个代码在我的电脑上失效了(别人那里好像是可以的)。。。不知道什么原因,selenium打开浏览器之后,无论是代码滑动滑块还是我手动滑动滑块,都是被判无效的,但是我自己打开浏览器操作是可以的。我觉得我可能是被针对了(以前也碰到过两次),感觉没有办法解决之后,我开始着手用其他方式解决自动登录——Cookie。其实吧,这个是套路,直接贴代码就行了,有一个地方需要注意注意注意,重要的事情说三遍,因为网上的教程在这个地方很多都是错的,从而导致我当时代码调得差点自闭了。
      def get_cookie(self):
         self.driver.get(damai_url)
         print("###请点击登录###")
         while self.driver.title.find('大麦网-全球演出赛事官方购票平台')!=-1:
           sleep(1)
         print("###请扫码登录###")
         while self.driver.title=='中文登录':
           sleep(1)
         print("###扫码成功###")
         pickle.dump(self.driver.get_cookies(), open("cookies.pkl", "wb"))
         print("###Cookie保存成功###")

      def set_cookie(self):
        try:
              cookies = pickle.load(open("cookies.pkl", "rb"))#载入cookie
              for cookie in cookies:
                  cookie_dict = {
                    'domain':'.damai.cn',#必须有,不然就是假登录
                    'name': cookie.get('name'),
                    'value': cookie.get('value'),
                    "expires": "",
                    'path': '/',
                    'httpOnly': False,
                    'HostOnly': False,
                    'Secure': False}
                  self.driver.add_cookie(cookie_dict)
              print('###载入Cookie###')
        except Exception as e:
              print(e)


  之后只要如此调用就可以了。
  if not os.path.exists('cookies.pkl'):#如果不存在cookie.pkl,就获取一下
     self.get_cookie()
  else:
     self.driver.get(damai_url)
     self.set_cookie()


  2.选择演唱会
  接下来就很直接了。
http://www.51testing.com/attachments/2023/04/15326880_202304241321331vBPE.jpg
  self.driver.find_elements_by_xpath('/html/body/div/div/div/input').send_keys(self.name)#找到搜索栏,填入演唱会歌星的名字
  self.driver.find_elements_by_xpath('/html/body/div/div/div/div').click()#点击旁边的搜索按钮


http://www.51testing.com/attachments/2023/04/15326880_202304241321361yYJP.jpg
  kinds=self.driver.find_element_by_id('category_filter_id').find_elements_by_tag_name('li')#选择演唱会类别
  for k in kinds:
     if k.text=='演唱会':
        k.click()
        break
  lists=self.driver.find_elements_by_id('content_list').find_elements_by_tag_name('li')#获取所有可能演唱会
  titles=[]
  links=[]
  self.choose_result=0
  for li in lists:
     word_link=li.find_element_by_tag_name('h3')
     titles.append(word_link.text)
     temp_s=word_link.get_attribute('innerHTML').find('href')+6
     temp_e=word_link.get_attribute('innerHTML').find('target')-2
     links.append(word_link.get_attribute('innerHTML'))
     if li.find_element_by_tag_name('h3').text.find(self.place)!=-1:#选择地点正确的演唱会
        self.choose_result=len(titles)
        break
  self.url="https:"+links
  self.driver.get(self.url)#载入至购买界面


  3.购票
http://www.51testing.com/attachments/2023/04/15326880_202304241321401v3Ey.jpg
  datelist=self.driver.find_element_by_id("performList").find_elements_by_tag_name('li')#根据优先级选择一个可行日期
  for i in self.date:
  j=datelist.get_attribute('class')
  if j=='itm':
  datelist.click()
  break
  elif j=='itm itm-sel':
  break
  elif j=='itm itm-oos':
  continue
  sleep(1)
  pricelist=self.driver.find_element_by_id("priceList").find_elements_by_tag_name('li')#根据优先级选择一个可行票价
  for i in self.price:
  j=pricelist.get_attribute('class')
  if j=='itm':
  pricelist.click()
  break
  elif j=='itm itm-sel':
  break
  elif j=='itm itm-oos':
  continue
  sleep(1.5)
  cart=self.driver.find_element_by_id('cartList')
  try:#各种按钮的点击
  try:
  cart.find_element_by_class_name('ops').find_element_by_link_text("立即预定").click()
  self.status=3
  except:
  cart.find_element_by_class_name('ops').find_element_by_link_text("立即购买").click()
  self.status=4
  except:
  cart.find_element_by_class_name('ops').find_element_by_link_text("选座购买").click()
  self.status=5
  self.num+=1
  sleep(0.5)


  接下来就是要点击确定了,但是这个确定按钮有三类:立即预定、立即购买、选座购买,我们需要分别处理。如果出错,就刷新界面
  cart=self.driver.find_element_by_id('cartList')
  try:#各种按钮的点击
  try:
  cart.find_element_by_class_name('ops').find_element_by_link_text("立即预定").click()
  self.status=3
  except:
  cart.find_element_by_class_name('ops').find_element_by_link_text("立即购买").click()
  self.status=4
  except:
  cart.find_element_by_class_name('ops').find_element_by_link_text("选座购买").click()
  self.status=5


  如果没有成功跳转,我没抢票的经验,不知道接下来会出现什么,所以也不知道要怎么具体应对,所以就刷新页面,重复上述操作,只能暂时这么粗暴地来了。
  4.确认订单
  点击选座购买,如果跳转成功,接下来就是选座了,这个有点小麻烦,暂时没写代码应对,开始交给人工处理。
  点击立即预定和立即购买后如果正常跳转,之后就是确认订单信息与结算了,这时会有好多类型的界面,但是其中两类界面占了大多数,所以我就针对这两类进行编程。
  因为这个时候的界面涉及很多隐私信息,我就不放出来了,虽然可以打码,但是我就是不想放。大致的操作其实和前面是类似的:找到目标,点击。
  print('###开始确认订单###')
  print('###默认购票人信息###')
  rn_button=self.driver.find_elements_by_xpath('/html/body/div/div/div/div/div/a')
  if len(rn_button)==1:#如果要求实名制
  print('###选择实名制信息###')
  rn_button.click()
  #选择实名信息
  tb=self.driver.find_element_by_xpath('/html/body/div/div/div/div')
  lb=tb.find_elements_by_tag_name('label')#选择第self.real_name个实名者
  lb.find_elements_by_tag_name('td').click()
  tb.find_element_by_class_name('one-btn').click()
  print('###默认选择付款方式###')
  print('###确认商品清单###')
  rn_button=self.driver.find_elements_by_xpath('/html/body/div/div/div/div/div/div/div/h2/a')
  if len(rn_button)==1:#如果要求实名制
  print('###选择购票人信息###')
  rn_button.click()
  #选择实名信息
  tb=self.driver.find_element_by_xpath('/html/body/div/div/div/div')
  lb=tb.find_elements_by_tag_name('label')#选择第self.real_name个实名者
  lb.find_elements_by_tag_name('td').click()
  tb.find_element_by_class_name('one-btn').click()
  print('###不选择订单优惠###')
  print('###请在付款完成后下载大麦APP进入订单详情页申请开具###')
  self.driver.find_element_by_id('orderConfirmSubmit').click()#同意以上协议并提交订单
  sleep(8)
  if self.driver.title.find('支付')!=-1:
  self.status=6
  print('###成功提交订单,请手动支付###')
  else:
  print('###提交订单失败,请查看问题###')


  我试了几个,基本都是可以抢票成功的,但是这是没有一窝蜂地去抢的情况下。

赵佳乐SMILE 发表于 2023-5-6 09:20:54

早看到,就能抢到周杰伦演唱会的票了
页: [1]
查看完整版本: 大麦网演唱会抢票程序之selenium