武林秘籍之python接口测试

Posted 隧唐技术部

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了武林秘籍之python接口测试相关的知识,希望对你有一定的参考价值。

       在前期工作中一直主要做UI层自动化测试,在分层自动化概念的影响下开始进行接口测试学习,现在接口测试工具主要为jmeter和soapUI,两款工具学习周期过长,故而小灰灰这里选择了python27+requests2.18+

unittest进行接口测试,因为python提供了强大的requests库,简洁明了,下面小灰灰带来一次接口测试实例。

接口测试流程

 模拟客户端连接服务器(服务器提供的端口是否可访问)

                         ↓
     客户端发送报文请求
         ↓
    服务器端接收请求并做处理
         ↓
   检查返回的预期结果并与实际结果对比
         ↓
         结束


测试思路

只要一步一步的走下来就行了。

数据源

       

数据源我选择使用的是JSON来保存,当然,也可以使用Excel来保存,由于我懒得去读取excel,所以选择用JSON来保存,同时也是因为JSON用起来比较方便, Python对JSON也提供了对应的库供使用。

{

  "TestId": "testcase004",

  "Method": "post",

  "Title": "单独推送消息",

  "Desc": "单独推送消息",

  "Url": "http://xxx.xxx.xxx.xx",

  "InputArg": {

   "action": "123456",

   "account": "18112345678",

   "uniqueid": "00D7C889-06A0-426E-BAB1-5741A1192038",

   "title": "测试测试",

   "summary": "拉拉啦",

   "message": "12345",

   "msgtype": "25",

   "menuid": "203"

  },

  "Result": {

   "errorno": "0"

  }

 }


      发送请求

发送请求相当来说是比较简单了,在python中用requests模块,然后从JSON中读取发送的参数,post、get都是可以的。因为后面还要生成测试报告,那么发送的数据时需要做一下记录,这里选择用txt文本来作为记录的容器。

f = file("case.json")

testData = json.load(f)

f.close()

def sendData(testData, num):

  payload = {}

  # 从json中获取发送参数

  for x in testData[num]['InputArg'].items():

    payload[x[0]] = x[1]

  with open('leftside.txt', 'a+') as f:

    f.write(testData[num]['TestId'])

    f.write('-')

    f.write(testData[num]['Title'])

    f.write('\n')

  # 发送请求

  data = requests.get(testData[num]['Url'], params=payload)

  r = data.json()


      接受返回

和发送请求时一样返回的数据也需要进行一次存储,当然选择用数据库存储,我觉得数据库存储太麻烦了,所以我选择用txt文本作为存储容器。

with open('rightside.txt', 'a+') as rs:

    rs.write('发送数据')

    rs.write('|')

    rs.write('标题:'+testData[num]['Title'])

    rs.write('|')

    rs.write('发送方式:'+testData[num]['Method'])

    rs.write('|')

    rs.write('案例描述:'+testData[num]['Desc'])

    rs.write('|')

    rs.write('|')

    rs.write('发送参数:'+str(payload).decode("unicode-escape").encode("utf-8").replace("u\'","\'"))

    rs.write('|')

    rs.write(testData[num]['TestId'])

    rs.write('\n')


结果判断

        结果判定这里使用全等于判定,当然也要根据实际接口的情况决定,也可以写成正则判定,判断的结果我定义了成功、失败或者没结果,结果的判断就看自己的需求进行定义了。

with open('result.txt', 'a+') as rst:

    rst.write('返回数据')

    rst.write('|')

    for x, y in r.items():

      rst.write(' : '.join([x, y]))

      rst.write('|')

    # 写测试结果

    try:

      if cmp(r, testData[num]['Result']) == 0:

        rst.write('pass')

      else:

        rst.write('fail')

    except Exception:

      rst.write('no except result')

    rst.write('\n')

我这里结果有3种,成功、失败或者没结果。结果的设置就看自己的定义了。

生成测试报告

        在自动化测试过程中测试报告往往是一个重头戏,由于前面我在发送数据、返回数据和结果的保存都是用txt文本存储,所以每次使用a+模式新增,会让结果越来越多,导致检查起来非常恼火,所以后面还是建议使用excel和数据库保存。

这里我的处理方式是每次测试完毕之后,用Python读取txt文本中的数据,然后使用Django动态生成一个结果,然后再使用requests抓取这个网页,保存在Report文件夹中。

网页报告

        这里对于Django的方法就不多说了,现在只需要在views文件中打开之前记录的3个txt文件,然后进行数据处理,返回给前端,前端用Bootstrap来渲染,就能生成一个比较漂亮的测试报告。

def index(request):

  rightside = []

  result = []

  rst_data = []

  leftside = []

  passed = 0

  fail = 0

  noresult = 0

  with open(os.getcwd() + '/PortTest/leftside.txt') as ls:

    for x in ls.readlines():

      lf_data = {

        'code': x.strip().split('-')[0],

        'title': x.strip().split('-')[1]

      }

      leftside.append(lf_data)

 

  with open(os.getcwd() + '/PortTest/rightside.txt') as rs:

    for x in rs.readlines():

      row = x.strip().split('|')

      rs_data = {

        "fssj": row[0],

        "csbt": row[1],

        "fsfs": row[2],

        "alms": row[3],

        "fsdz": row[4],

        "fscs": row[5],

        'testid': row[6]

      }

      rightside.append(rs_data)

 

  with open(os.getcwd() + '/PortTest/result.txt') as rst:

    for x in rst.readlines():

      row = x.strip().split('|')

      if row[len(row)-1] == 'fail':

        fail += 1

      elif row[len(row)-1] == 'pass':

        passed += 1

      elif row[len(row)-1] == 'no except result':

        noresult += 1

 

      rs_data = []

      for y in row:

        rs_data.append(y)

      result.append(rs_data)

  for a, b in zip(rightside, result):

    data = {

      "sendData": a,

      "dealData": b,

      "result": b[len(b)-1]

    }

    rst_data.append(data)

  return render(request, 'PortTest/index.html', {"leftside": leftside,

                          "rst_data": rst_data,

                          "pass": passed,

                          "fail": fail,

                          "noresult": noresult})

基本上这里用到的都是一些python很基础的知识,字符串分割等等。所以小灰灰在这里强调下:想做自动化或者接口测试,一定要掌握好python!这里数据处理为了方便,在获取数据存储的时候就要按照一定的格式来存储,views的方法就很容易做处理。


      前端代码

        下面是我这次接口测试报告前端的代码,当然网上也有很多测试老鸟自己写的测试报告页面文件,可以根据自己的需求进行改良。

<!DOCTYPE html>

<html>

<head>

  <meta charset="UTF-8">

  <title>Title</title>

  <link href="http://jb51.net/bootstrap/3.0.3/css/bootstrap.min.css" rel="stylesheet">

  <script ></script>

  <script ></script>

</head>

<body>

<div>

  <div>

    <div>

      <h1>接口测试报告

        <small>Design By Sven</small>

      </h1>

    </div>

  </div>

  <div>

    <div>

      <h3>测试通过 <span class="label label-success">{{ pass }}</span></h3>

    </div>

    <div>

      <h3>测试失败 <span class="label label-danger">{{ fail }}</span></h3>

    </div>

    <div>

      <h3>无结果 <span class="label label-warning">{{ noresult }}</span></h3>

    </div>

  </div>

  <p></p>

  <div>

    <div>

      <ul>

        {% for ls in leftside %}

          <li><a href="#{{ ls.code }}">{{ ls.code }} - {{ ls.title }}</a></li>

        {% endfor %}

      </ul>

    </div>

    <div>

      {{ x.result }}

      {% for x in rst_data %}

        <div id="accordion">

        {% if x.result == 'pass' %}

          <div class="panel panel-success">

        {% elif x.result == 'fail' %}

          <div class="panel panel-danger">

        {% elif x.result == 'no except result' %}

          <div class="panel panel-warning">

        {% endif %}

 

      <div>

        <h4>

          <a data-toggle="collapse" href="#{{ x.sendData.testid }}">

            {{ x.sendData.testid }} - {{ x.sendData.csbt }}

          </a>

        </h4>

      </div>

      <div id="{{ x.sendData.testid }}" class="panel-collapse collapse">

        <div>

          <b>{{ x.sendData.fssj }}</b><br>

          {{ x.sendData.csbt }}<br>

          {{ x.sendData.fsfs }}<br>

          {{ x.sendData.alms }}<br>

          {{ x.sendData.fsdz }}<br>

          {{ x.sendData.fscs }}

          <hr>

          {% for v in x.dealData %}

            {{ v }}<br>

          {% endfor %}

        </div>

      </div>

      </div>

      </div>

        <p></p>

      {% endfor %}

      </div>

      </div>

    </div>

    <script>

      $(function () {

        $(window).scroll(function () {

          if ($(this).scrollTop() != 0) {

            $("#toTop").fadeIn();

          } else {

            $("#toTop").fadeOut();

          }

        });

        $("body").append("<div id=\"toTop\" style=\"border:1px solid #444;background:#333;color:#fff;text-align:center;padding:10px 13px 7px 13px;position:fixed;bottom:10px;right:10px;cursor:pointer;display:none;font-family:verdana;font-size:22px;\">^</div>");

        $("#toTop").click(function () {

          $("body,html").animate({scrollTop: 0}, 800);

        });

      });

    </script>

</body>

</html>


      测试报告效果图

      总结

      本菜鸟认为软件测试是一个需要不断学习、不断提升个人能力的行业,自动化测试的学习也好、测试理论的学习也好,只要还在学习,只要还能看到自己和他人的差距,就要去学习、练习,和行业中人多交流!软件测试人员个人能力的迭代和软件的迭代一样,都是一个不断追求卓越的过程!


以上是关于武林秘籍之python接口测试的主要内容,如果未能解决你的问题,请参考以下文章

Python接口自动化之登录接口测试

Python接口自动化之登录接口测试

Python接口自动化之unittest单元测试

Python接口自动化测试之pytest与unittest区别

Python接口自动化测试之pytest与unittest区别

软件测试进阶之自动化测试——python接口测试实例