当前位置:网站首页>Use pytest hook function to realize automatic test result push enterprise WeChat

Use pytest hook function to realize automatic test result push enterprise WeChat

2022-08-04 10:33:00 51CTO

前言


通常,After the completion of the automated test cases in the execution,Send a result,To notify the tester or testleader测试的结果.If you have any testing failure,The tester to check the specific test report,Check what scene is no test to pass.The more popular ways there are to remind:

  • 邮件
  • 企业微信、钉钉等push消息

Because our company office software is used by enterprise WeChat,因此,In the realization of function of test results notifications when,Selection is the enterprise WeChat.There are two forms of the realization of the more popular way:

  • Enterprise WeChat application notice:Need to create an application in enterprise WeChat,再获取Secret
  • Ordinary group of news to push:Need to add in the group of a group of robots(会自动生成webhook_url,For later interface call)

Because of the way a need to create application in enterprise WeChat(Need administrator permissions operation),Overall implement more complicated,So I choose is the second population robot实现方式.


一、Implementation principle and implementation effect

1.External link process

利用pytest hookFunction to push enterprises WeChat automated test result_自动化测试

2.Inside the principle and process

利用pytest hookFunction to push enterprises WeChat automated test result_pytest hook_02

1)各模块&方法功能:
  • RedisHandler基类:用于初始化redis连接、查询数据、写入数据
  • CaseCount基类:For initial use case statistics、To obtain statistical success&失败&跳过&An error with the number of cases、Pass rate calculation cases
  • EnterpriseWechatNotification基类:Used to define the content of the enterprise WeChat messages sent template、定义调用hook_url(群机器人)发送消息方法
  • hook方法pytest_runtest_makereport:用于获取pytestAfter execution of test results、将结果写入缓存、Generate the console test report
  • hook方法pytest_terminal_summary:用于获取执行结果、Call sends the message method WeChat messages sent
2)具体调用原理、流程:

① 前提:

  • Have add enterprise WeChat group of robot,并记住hook地址;
  • python+pytestHas to write the test case;

利用pytest hookFunction to push enterprises WeChat automated test result_自动化测试_03

② pytest运行测试用例,RedisHandler连接redis                                     ,pytest_runtest_makereport获取用例执行结果,并:

  • 调用RedisHandlerThe write cache method,将结果写入缓存;
  • 调用CaseCountThe acquisition cases to pass the calculation cases pass method;
  • Will get to every test results of the output to the console to display:↓(Windows本地运行效果)

利用pytest hookFunction to push enterprises WeChat automated test result_Jenkins_04

③ pytest_terminal_summary方法:

  • 分别调用CaseCountGet through in、失败、跳过、The number of error of use case method(此方法调用RedisHandler中的get_key方法),获取到各个(通过、失败、跳过、报错)执行结果统计;
  • 调用CaseCountThe acquisition cases to pass the calculation cases pass method;
  • 调用EnterpriseWechatNotificationThe method of enterprise WeChat messages sent,To get to all(通过、失败、跳过、报错)Results the number of statistics andEnterpriseWechatNotificationThe predefined templates for Mosaic,发送到企业微信;
  • To get to all(通过、失败、跳过、报错)The results together with the cases to pass,The output to the console display:↓(Windows本地运行效果)

利用pytest hookFunction to push enterprises WeChat automated test result_Jenkins_05

二、编码实现

1.各个基类

RedisHandler基类:

      
      
import redis


class RedisHandler:
def __init__(self, host, port=6379, db=0):
self.client = redis.StrictRedis(host=host, port=port, db=db) # 生成客户端连接,StrictRedis()Default to the connection pool,No longer aloneConnectPool

def set_string(self, name: str, value, ex=None, px=None, nx=False, xx=False) -> None:
"""
缓存中写入str(单个)
:param name: 缓存名称
:param value: 缓存值
:param ex: 过期时间(秒)
:param px: 过期时间(毫秒)
:param nx: 如果设置为True,则只有name不存在时,当前set操作才执行(新增)
:param xx: 如果设置为True,则只有name不存在时,当前set操作才执行(修改)
:return:
"""
self.client.set(name, value=value, ex=ex, px=px, nx=nx, xx=xx)

def incr(self, key):
"""
使用incr方法,处理并发问题
当key不存在时,Will first initial as0,每次调用,则会+1
:param key:
:return:
"""
self.client.incr(key)

def get_key(self, name):
"""读取缓存"""
result = self.client.get(name)
return result
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.
  • 33.


CaseCount基类:

      
      
from api_test.common.redis_handler import RedisHandler
from api_test.config.db_config import DBConfig


class CaseCountName:
ERROR: str = "error_count"
FAILED: str = "failed_count"
PASSED: str = "passed_count"
SKIP: str = "skip_count"
TOTAL: str = "total_count"


class CaseCount:
"""
redis 缓存统计用例执行情况
"""

def __init__(self):
self.redis = RedisHandler(host=DBConfig.redis_config.get('host')) # redisThe host address can write die here,Can also be obtained from the configuration class

def init_process(self):
"""
初始化进度、总数、成功数、失败数
"""
self.redis.set_string(CaseCountName.TOTAL, 0)
self.redis.set_string(CaseCountName.SKIP, 0)
self.redis.set_string(CaseCountName.PASSED, 0)
self.redis.set_string(CaseCountName.FAILED, 0)
self.redis.set_string(CaseCountName.ERROR, 0)

def failed_count(self):
"""失败用例数"""
return int(self.redis.get_key(CaseCountName.FAILED))

def passed_count(self):
"""By the total number of cases"""
return int(self.redis.get_key(CaseCountName.PASSED))

def skip_count(self):
"""跳过用例数"""
return int(self.redis.get_key(CaseCountName.SKIP))

def error_count(self):
"""An error with the number of cases"""
return int(self.redis.get_key(CaseCountName.ERROR))

def total_count(self):
"""用例总数"""
return int(self.redis.get_key(CaseCountName.TOTAL))

def pass_rate(self):
"""用例成功率"""
try:
rate = round((self.passed_count() + self.skip_count()) / self.total_count() * 100, 2)
return rate
except ZeroDivisionError:
raise Exception("执行失败,Not detected cases to perform the number")
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.
  • 33.
  • 34.
  • 35.
  • 36.
  • 37.
  • 38.
  • 39.
  • 40.
  • 41.
  • 42.
  • 43.
  • 44.
  • 45.
  • 46.
  • 47.
  • 48.
  • 49.
  • 50.
  • 51.
  • 52.
  • 53.
  • 54.
  • 55.
  • 56.
  • 57.


EnterpriseWechatNotification基类:

      
      
import os
import json
import requests
import platform


def get_env_from_jenkins(name, base=''):
"""从JenkinsIn the global environment variables"""
return os.getenv(name) and os.getenv(name).strip() or base


ProjectName = get_env_from_jenkins("JOB_NAME") # Jenkins构建项目名称
BUILD_URL = get_env_from_jenkins("BUILD_URL") # Jenkins构建项目URL
BUILD_NUMBER = get_env_from_jenkins("BUILD_NUMBER") # Jenkins构建编号


class EnterpriseWechatNotification:
def __init__(self, hook: list):
# Enterprise WeChat group of robothook地址,A robot is a,More than just define multiple,可以写死,Can also be written in a configuration class
self.hook_url_list = [f"https://qyapi.weixin.qq.com/cgi-bin/webhook/send?key={i}" for i in hook]
# allureGenerate reports address,JenkinsExecution will use,Windows暂未配置allure地址
self.allure_url = f"http://192.168.1.122:8088/jenkins/job/{ProjectName}/{BUILD_NUMBER}/allure/"
self.header = {'Content-Type': 'application/json'}

def send_msg(self, result=''):
"""发送企业微信消息通知"""
global payload
linux_content = f"""** 【{ProjectName}】**
> 项目名称:{ProjectName}
> Member number:#{BUILD_NUMBER}
> 测试环境:{platform.system()}
> [报告链接]({self.allure_url})
> [控制台链接]({BUILD_URL})
{result}"""

windows_content = f"""** 【auto_test_project】**
> 测试环境:{platform.system()}
{result}"""
if platform.system() == "Linux":
payload = {
"msgtype": "markdown",
"markdown": {
"content": linux_content
}
}
elif platform.system() == "Windows":
payload = {
"msgtype": "markdown",
"markdown": {
"content": windows_content
}
}
for hook_url in self.hook_url_list:
requests.post(url=hook_url, headers=self.header, data=json.dumps(payload))
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.
  • 33.
  • 34.
  • 35.
  • 36.
  • 37.
  • 38.
  • 39.
  • 40.
  • 41.
  • 42.
  • 43.
  • 44.
  • 45.
  • 46.
  • 47.
  • 48.
  • 49.
  • 50.
  • 51.
  • 52.
  • 53.
  • 54.

注意事项: get_env_from_jenkins方法为从Jenkins获取全局变量,Check the path of the global variables as:Jenkins流水线语法-全局变量-env,见下图:

利用pytest hookFunction to push enterprises WeChat automated test result_Jenkins_06

2.pytest的hook方法,定义在conftest.py中

      
      
import time
import pytest
from api_test.config.config import HookUrlConf
from api_test.common.send_enterprise_wechat import EnterpriseWechatNotification
from api_test.common.redis_handler import RedisHandler
from api_test.config.db_config import DBConfig
from api_test.common.case_count_control import CaseCountName, CaseCount

redis = RedisHandler(host=DBConfig.redis_config.get('host'))
case_count = CaseCount()
case_count.init_process() # 初始化RedisUse cases in cache data


@pytest.hookimpl(hookwrapper=True, tryfirst=True)
def pytest_runtest_makereport(item, call):
"""获取测试结果、生成测试报告"""
print('------------------------------------')
out = yield
report = out.get_result()
if report.when == 'call':
# print(f"测试报告:{report}")
# print(f"步骤:{report.when}")
print(f"用例id:{report.nodeid}")
print(f"用例描述:{str(item.function.__doc__)}")
print(f"运行结果:{report.outcome}")
"""Will write cache case execution results"""
if report.outcome == 'passed':
redis.incr(CaseCountName.PASSED)
if report.outcome == 'failed':
redis.incr(CaseCountName.FAILED)

if report.when == 'setup':
if report.outcome == 'skipped':
redis.incr(CaseCountName.SKIP)

CaseCount().total_run_count()


def pytest_terminal_summary(terminalreporter, exitstatus, config):
"""收集测试结果,从RedisCache data to derive"""
total_case = case_count.total_count()
pass_case = case_count.passed_count()
fail_case = case_count.failed_count()
skip_case = case_count.skip_count()
error_case = case_count.error_count()
pass_rate = case_count.pass_rate()
run_time = round((time.time() - terminalreporter._sessionstarttime), 2)
print("******Use case execution result statistics******")
print(f"总用例数:{total_case}条")
print(f"通过:{pass_case}条")
print(f"失败:{fail_case}条")
print(f"跳过:{skip_case}条")
print(f"报错:{error_case}条")
print(f"用例通过率:{pass_rate}%")
print(f"用时:{run_time}s")
desc = """
This implementation is as follows:
Always use cases as:{}
通过用例数:<font color=\"info\">{}条</font>
失败用例数:<font color=\"warning\">{}条</font>
Error with the number of cases:{}
跳过用例数:{}
通过率为:{} %
用时:{}s
""".format(total_case, pass_case, fail_case, error_case, skip_case, pass_rate, run_time)
EnterpriseWechatNotification(hook=HookUrlConf.HOOK_URL.value).send_msg(desc) # The results sent enterprise WeChat
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.
  • 33.
  • 34.
  • 35.
  • 36.
  • 37.
  • 38.
  • 39.
  • 40.
  • 41.
  • 42.
  • 43.
  • 44.
  • 45.
  • 46.
  • 47.
  • 48.
  • 49.
  • 50.
  • 51.
  • 52.
  • 53.
  • 54.
  • 55.
  • 56.
  • 57.
  • 58.
  • 59.
  • 60.
  • 61.
  • 62.
  • 63.
  • 64.
  • 65.
  • 66.


三、Operation process and operation effect

1.运行过程

  • Windows本地运行
  • Jenkins触发运行


2.Enterprise WeChat receives the test results inform:

  • 通过JenkinsTrigger a run notice effect:↓

利用pytest hookFunction to push enterprises WeChat automated test result_Jenkins_07

  • WindowsLocal manual trigger operation notice effect:↓

利用pytest hookFunction to push enterprises WeChat automated test result_Jenkins_08

小结


以上就是利用pytest的hook函数(pytest_runtest_makereport、pytest_terminal_summary‍)+redis,Implement WeChat send test results to the enterprise the principle and process of,And, of course, some disadvantages,如:

  • Whether the interface test automation orUIAutomated tests can be in this way to implement notifications;
  • In addition to the call in your codepytest hookFunction realization of news,JenkinsAlso can be achieved by installing a plug-in email notification、执行PythonScript to achieve the goal of enterprise micro message notice;
  • The stored test results may not want to useredis,Can also be written in local files, etc,Because more than one layer calls,Just one more layer handles and could face an error,另外redisThe server connection error will affect the normal operation of the case;
  • Send the message content style supportMarkdown,Send content can also continue to optimize,比如:Notice what cases to an error, and so on;

更多实用干货,同步首发于微信公众号【测试开发实战】,欢迎关注!

利用pytest hookFunction to push enterprises WeChat automated test result_自动化测试_09

原网站

版权声明
本文为[51CTO]所创,转载请带上原文链接,感谢
https://yzsam.com/2022/216/202208041015361374.html