1. 背景

前段时间发现一个论坛上(https://npupt.com/blackjack.php)有21点小游戏。

这个21点小游戏的规则是每个人开局都会获得随机点数,如果觉得点数小,可以继续摸牌。如果摸牌后点数大于21点,系统会自动结束摸牌,否则可以继续摸牌。系统会自动对比2个参与者的点数,点数大的人获胜。(大于21点会被视为1点)。

由于整个游戏过程都是通过鼠标点击网页来完成的,那么我们可以让代码帮我们完成自动玩这个小游戏。

2. 理清游戏逻辑

如上图,点击“开始游戏”后,系统会给我们随机点数(如20点)。然后我们可以选择“再抓一张”或者“不要抓了,结束”。但是在这个例子中,20点已经很大了,我们再抓牌,很容易就超过21点,那就得不偿失了,故这里我选择了“不要抓了,结束”。(如果选择“再抓一张”,结果画面和上图的右侧相同。)

如上图,当选择结束游戏后,系统就会等待对手完成。只有对手完成了,我们才能开始新的21点小游戏。

上述便是这个小游戏的玩法。

顺便,总结下21点小游戏的用户操作,主要有4个:

  1. 开始游戏;
  2. 再抓一张;
  3. 结束游戏;
  4. 等待对手完成。

3. 游戏逻辑转换成脚本思路

既然我们知道了玩法,那么我们只需将这些玩法用计算机思维将其转换成脚本可以执行的步骤就行。

上述玩法可以转换成以下脚本思路:

  1. 等待开局。
  2. 开始21点。
  3. 判断点数。
  4. 若点数大于20,回到第1步。
  5. 若点数大于17,则去到第8步。(17可变,只是我认为17点够大了)
  6. 若点数小于等于17,则去到第7步。
  7. 继续摸牌,回到第3步。
  8. 终止游戏,回到第1步。

4. 思考如何用python实现

对于接触过html和学过计算机网络的同学,可能第一时间就会知道关键在于“网络请求”

一切的网页操作都是需要和服务器交互的,所以只要捕获到这些操作的“网络请求”,我们就可以知道某个操作中浏览器请求的服务器,和请求的内容,以及服务器返回的内容。

而python具备向服务器发出请求的能力,如requests库,我们只需用requests库模拟用户向服务器发出请求即可达到模拟用户操作的目的。

故,用python实现自动玩这个小游戏,需要3大步骤:

  1. 使用抓包工具,监听在“理清游戏逻辑”中提到的4大操作的网络请求
  2. 然后用python模拟网络请求。
  3. 用python实现游戏逻辑。

5. 准备工作

  • 安装好Python 2.7。

    (Linux或Mac下的安装教程https://www.cnblogs.com/toulanboy/p/7753502.html)

  • 通过pip安装requests库。

    pip install requests
    
  • 安装谷歌浏览器 (可以用来抓包,捕获网络请求)

6. 捕获网络请求

谷歌浏览器有一个调试工具,这个调试工具可以监听浏览器的一切行为,包括了我们最想知道的“网络请求”。下面讲述捕获21点小游戏网络请求的具体过程。

  1. 打开谷歌浏览器,输入网址:https://npupt.com/blackjack.php。

  2. 然后对页面右击,选择“检查”,就会出现下图的界面。左侧是正常网页,右侧便是我们要用到的工具。

  3. 然后点击“抓包工具”界面的“network”

  4. 点击左侧的“开始游戏”。会得到下图界面。包含以下信息:

    • 左侧网页可以知道,我一开始是16点。
    • 右侧网页捕获到了“开始游戏”这个操作的网络请求。

  5. 那么,我们点击上图中“关键页面”所标记的网页,然后在最右侧栏可以得到以下信息。

  6. 上述便是网络请求。包含了请求信息/服务器返回信息。上面的信息太多,我提取了一下, 对我们最有用的是以下内容。

   Request URL(请求的服务器): https://npupt.com/blackjack.php
   Request Method(请求方式): POST
//请求参数 - toulanboy - http://www.cnblogs.com/toulanboy/
game: hit
start: yes

总结本节,通过这个过程,我们知道了“开始游戏”的请求内容,那么同理,“再抓一张”,“结束游戏”也可以得到相应内容。而,”等待对手结束“则需要检查网页源代码,只需对网页关键位置右键,选择“检查”,即可得到该位置的源代码。这些都是为python脚本准备的关键素材。最后每个操作需要的关键信息如下:

1. 等待对手结束

若之前的21点还没有结束(暂时没有对手上线),那么不能开局,需等待之前的结束。若需等待,主页面包含以下内容

<button type="submit" class="btn btn-default">刷新</button>

2. 开始游戏

若可以开始游戏,主页面包含以下内容

<button type="submit" class="btn btn-primary">开始游戏!</button>

3. 开始21点小游戏

向主页面(https://npupt.com/blackjack.php)post数据

game: hit,
start: yes

4. 判断点数

判断每次操作后,主页面返回的网页内容。点数的html样式如下:

<b>点数 = 16</b>

5. 继续摸牌

向主页面post数据

game: hit

6. 结束游戏

向主页面post数据

game: stop

7. 用python模拟网络请求

模拟请求的思路是:通过Python的网络库requests,向指定服务器发送指定参数即可。具体实现如下:

模拟Get请求,判断是否可以开始游戏

#function - Get网页
#若网页显示之前的没结束,则返回0
#若网页显示可以开始新的一局,则返回1
#其它情况返回-1
def getData(url):
    headers = {
        \'User-Agent\' : \'-\',#建议设置
        \'cookie\':\'-\'#我没做登陆,所以手动弄cookie
        }
    try:
        response = requests.get(url, headers=headers)
        indexHtmlCode = response.text
        indexHtmlCode =  indexHtmlCode.encode(\'utf-8\')
        
        #判断是否有刷新按钮,若有,说明上局没结束
        freshRegex = r\'刷新</button>\'
        result = re.findall(freshRegex, indexHtmlCode)
        if result:
            return 0
# toulanboy - http://www.cnblogs.com/toulanboy/        
        #判断是否有开始游戏按钮,若有,说明可以开始游戏
        beginRegex = r\'开始游戏!</button>\'
        result = re.findall(beginRegex, indexHtmlCode)
        if result:
            return 1

		return -1

    except Exception as e:
        return -1

toulanboy – http://www.cnblogs.com/toulanboy/

模拟Post请求,实现开始/摸牌/停止的用户操作**

### 不同动作需要的数据不一样
# 开始游戏
startValues = {
    "game":"hit",
    "start":"yes"
}
# 继续摸牌
hitValues = {
    "game":"hit"
}
# 停止摸牌
stopValues = {
    "game":"stop"
}

#function - Post网页
#若是开始和摸牌,则返回点数
#页面没有点数(停止摸牌操作会出现),则返回0
#异常,返回-1
def postData(url, values):
    dd = urllib.urlencode(values)
    headers = {
            "Content-Length":str(len(dd)),
            "Content-Type":"application/x-www-form-urlencoded",
            \'Cache-Control\':\'max-age=0\',#上述三个参数其实应该不用手动添加,有可能request库会帮我们添加。有待验证。
            \'User-Agent\' : \'-\',#自己补充
            \'cookie\':\'-\'#自己补充
            }
    try:
        response = requests.post(url, data=dd,headers=headers)
        indexHtmlCode = response.text
        indexHtmlCode =  indexHtmlCode.encode(\'utf-8\')
# toulanboy - http://www.cnblogs.com/toulanboy/
        # 查看返回的网页的点数
        pointRegex = r\'点数\s?=\s?(\d*)<\'
        result = re.findall(pointRegex, indexHtmlCode)
        if result:
            return int(result[0])
        else:
            return 0

    except Exception as e:
        return -1

8. 用python实现游戏逻辑


url = \'https://npupt.com/blackjack.php\' 

#不停地玩21点
while True:
    #先看之前的是否结束了
    result = getData(url)
    time.sleep(5) # toulanboy - http://www.cnblogs.com/toulanboy/
    if result == 0: #如果还没结束,则继续刷新
        print "之前的尚没结束,等待中"
    elif result == 1:#如果结束了,则开始游戏
        point = postData(url, startValues)#发出“开始游戏”请求
        print "已开局,当前点数 = %d" % point
        #大于20点,系统会自动结束,故在这里我只需在小于21点的情况下摸牌
        while point <= 20:
            if point >= 17:#我认为只要大于17点我满足了,所以大于17点就停止摸牌
                time.sleep(1)
                postData(url, stopValues)#发出“停止摸牌”请求
                print "停止摸牌了,当前点数 = %d" % point
                break
            else:#小于17点则继续摸牌
                time.sleep(1)
                point = postData(url, hitValues)#发出“继续摸牌”请求
                print "又摸牌了,当前点数 = %d" % point
             
        print "这局结束了,当前点数 = %d" % point
    else:#出现异常,则停止游戏。等待渣渣看日志看看哪里出问题了。
        sendEmail("xxx@qq.com", "Some errors occurred in python script for npubits", "Some errors occurred in python script for npubits")
        break

9. 完整代码

为了便于维护,完整代码中增加了日志记录和邮件提醒

#!/usr/bin/python
# coding=utf-8
# 时间:2018-08-22
# 作者:toulanboy
# 缘由:想实现自动玩npubits的21点游戏

import requests
import re
import urllib
import time 
import logging
import smtplib
from email.mime.text import MIMEText

#配置日志
logging.basicConfig(filename=\'my.log\',format=\'[%(asctime)s-%(filename)s-%(levelname)s:%(message)s]\', level = logging.INFO,filemode=\'a\',datefmt=\'%Y-%m-%d %I:%M:%S %p\')

### 不同动作需要的数据不一样
# 开始游戏
startValues = {
    "game":"hit",
    "start":"yes"
}
# 继续摸牌
hitValues = {
    "game":"hit"
}
# 停止摸牌
stopValues = {
    "game":"stop"
}
# toulanboy - http://www.cnblogs.com/toulanboy/
#function - Post网页
#若是开始和摸牌,则返回点数
#页面没有点数(停止摸牌操作会出现),则返回0
#异常,返回-1
def postData(url, values):
    dd = urllib.urlencode(values)
    headers = {
            "Content-Length":str(len(dd)),
            "Content-Type":"application/x-www-form-urlencoded",
            \'Cache-Control\':\'max-age=0\',#上述三个参数其实应该不用手动添加,有可能request库会帮我们添加。有待验证。
            \'User-Agent\' : \'-\',#自己补充
            \'cookie\':\'-\'#自己补充
            }
    try:
        response = requests.post(url, data=dd,headers=headers)
        indexHtmlCode = response.text
        indexHtmlCode =  indexHtmlCode.encode(\'utf-8\')
        # 提取网页主干,存入日志(方便后期的分析)
        body = re.findall(r\'<div\s?id=\\'main\\'\s?class=\\'well\s?no-border-radius\\'>.*?</div>\',indexHtmlCode, re.S)
        if body:
            logging.info(body[0])
        else:
            logging.info(indexHtmlCode)
        # 查看返回的网页的点数
        pointRegex = r\'点数\s?=\s?(\d*)<\'
        result = re.findall(pointRegex, indexHtmlCode)
        if result:
            return int(result[0])
        else:
            return 0

    except Exception as e:
        logging.error(e)
        return -1

#function - Get网页
#若网页显示之前的没结束,则返回0
#若网页显示可以开始新的一局,则返回1
#其它情况返回-1
def getData(url):
    headers = {
        \'User-Agent\' : \'-\',#建议设置
        \'cookie\':\'-\'#我没做登陆,所以手动弄cookie
        }
    try:
        response = requests.get(url, headers=headers)
        indexHtmlCode = response.text
        indexHtmlCode =  indexHtmlCode.encode(\'utf-8\')
        
        #判断是否有刷新按钮,若有,说明上局没结束
        freshRegex = r\'刷新</button>\'
        result = re.findall(freshRegex, indexHtmlCode)
        if result:
            return 0
        
        #判断是否有开始游戏按钮,若有,说明可以开始游戏
        beginRegex = r\'开始游戏!</button>\'
        result = re.findall(beginRegex, indexHtmlCode)
        if result:
            return 1

        # 若以上情况都不是,有可能是cookie过期了
        loginRegex = r\'您尚未登录</body>\'
        result = re.findall(loginRegex, indexHtmlCode)
        if result:
            return 2
        
        # 如果不是cookie过期,则需打印当前错误信息(记录返回的网页源代码),方便后面找错
        logging.error(indexHtmlCode)

    except Exception as e:
        logging.error(e)
        return -1

#发邮件 (收件人 ,邮件主题 ,邮件正文)
def sendEmail(_to, subject, mainText):
    _user = "-"  #登录的163邮箱
    _pwd = "-"   #163邮箱授权码
    msg = MIMEText(mainText)    #邮件正文
    msg["Subject"] = subject    #邮件主题
    msg["From"]    = _user      #发件人
    msg["To"]      = _to        #收件人

    try:
        s = smtplib.SMTP_SSL("smtp.163.com", 465)
        s.login(_user, _pwd)#登录
        s.sendmail(_user, _to, msg.as_string())#发送
        s.quit()#退出登录
        logging.info("邮件发送成功!")
        print "邮件发送成功!"
    except smtplib.SMTPException,e:
        logging.info("邮件发送失败,%s"%e[0])
        print "邮件发送失败,%s"%e[0]


url = \'https://npupt.com/blackjack.php\' 

#不停地玩21点
while True:
    #先看之前的是否结束了
    result = getData(url)
    time.sleep(5) 
    if result == 0: #如果还没结束,则继续刷新
        print "之前的尚没结束,等待中"
    elif result == 1:#如果结束了,则开始游戏
        point = postData(url, startValues)#发出“开始游戏”请求
        logging.info("已开局,当前点数 = %d" % point)
        print "已开局,当前点数 = %d" % point
        #大于20点,系统会自动结束,故在这里我只需在小于21点的情况下摸牌
        while point <= 20:
            if point >= 17:#我认为只要大于17点我满足了,所以大于17点就停止摸牌
                time.sleep(1)
                postData(url, stopValues)#发出“停止摸牌”请求
                logging.info("停止摸牌了,当前点数 = %d" % point)
                print "停止摸牌了,当前点数 = %d" % point
                break
            else:#小于17点则继续摸牌
                time.sleep(1)
                point = postData(url, hitValues)#发出“继续摸牌”请求
                logging.info("又摸牌了,当前点数 = %d" % point)
                print "又摸牌了,当前点数 = %d" % point
        
        logging.info("这局结束了,当前点数 = %d" % point)       
        print "这局结束了,当前点数 = %d" % point
    else:#出现异常,则停止游戏。等待渣渣看日志看看哪里出问题了。
        sendEmail("xxx@qq.com", "Some errors occurred in python script for npubits", "Some errors occurred in python script for npubits")
        break

toulanboy – http://www.cnblogs.com/toulanboy/

10 . 运行效果

从下图可知,运行正常。

版权声明:本文为toulanboy原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/toulanboy/p/9522848.html