背景是这样的:

这个组的测试人员每跑一个case都要上传测试结果附件到QC。每个待测功能模块可能包含几十上百的case。于是手工上传测试结果变成了繁重的体力劳动。令人惊讶的是我们的工具开发组竟然说做不了QC的测试结果附件上传。更让我惊讶的是,测试人员竟然真的手工上传结果上传了大半年了。

以下我写了个小工具解决这个问题:

思路很简单,调用hp提供的ALM Rest api接口。把一个个用户操作转化成http请求。然后按照接口要求,把附件一个一个上传到指定的QC test instance上。

主要用的库是:
requests:负责发送http请求
BeautifulSoup4:负责解析QC服务器返回的响应。
整个工具非常简单。

具体步骤如下:
先让用户登录进QC

[2017-05-17 13:57:25,023] Starting new HTTP connection (1): XXXXX.com
[2017-05-17 13:57:25,430] post http://XXXXX.com:80/qcbin/authentication-point/authenticate
[2017-05-17 13:57:25,430] headers = {'Authorization': 'Basic XXXXXXX'}
[2017-05-17 13:57:25,430] 200 OK
[2017-05-17 13:57:25,430] --------------------

(上面这个带了时间戳的东西是我的log。每个请求都会自己记下来发了些啥。)

然后按照用户给定的的test_set_id去QC里搜索一下这个test set下的test instance有哪些。

[2017-05-17 13:57:25,430] --------------------
[2017-05-17 13:57:26,303] get http://XXXXX.com:80/qcbin/rest/domains/{Domain name}/projects/{Project name}/test-instances?query={contains-test-set.id[XXXXXX]}&fields=id,test-id&page-size=2000
[2017-05-17 13:57:26,303] 200 OK
[2017-05-17 13:57:26,303] --------------------

然后每个test instance id去查一下qc里对应的test id,也就是测试用例的id,再根据这个id,从qc里查出测试用例的名字。
搜索的时候,这样指定返回值的字段名,以下是搜索某个test set id下的test-instance,并且要求返回值只包含test instance id和 test id
test-instances?query={contains-test-set.id[XXX]}&fields=id,test-id&page-size=2000

[2017-05-17 13:57:26,303] --------------------
[2017-05-17 13:57:26,516] get http://XXXXX.com:80/qcbin/rest/domains/XXX/projects/XXX/tests?query={id[XXXXXX]}&fields=name&page-size=2000
[2017-05-17 13:57:26,516] 200 OK
[2017-05-17 13:57:26,516] --------------------

根据上面查出来的test-id再次查询,得到这个test的name字段值

查出了测试用例的名字是用来和我要上传的附件做匹配的,这两个名字对上了,就会把某个附件传到这个用例对应的test-instance上。

上传文件部分调用的是这样的一个requests请求:

data = {
'filename': ("", attachment_name),
'override-existing-attachment':("","y")
} file = {
'file': open(attachmentpath+attachment_name, 'rb')
}
response = self.session.post(url, files=file,data=data)

最后附上一些代码,供参考:

这个类是用来做get请求和post请求的。auto_log装饰器会把发了什么请求打成log。
不像某些QC操作的库里一样满屏的打log

import logging as log
import requests class RestClient:
def __init__(self):
self.headers={}
self.session = requests.Session()
log.basicConfig(level=log.INFO, format='[%(asctime)s] %(message)s') def auto_log(func):
def wrapper(*args, **kw):
r = func(*args, **kw)
log.info("%s %s", func.func_name, args[1])
for key in kw:
log.info("%s = %s",key, kw[key])
log.info("%s %s",r.status_code,r.reason)
log.info("--------------------")
return r
return wrapper @auto_log
def get(self, url):
return self.session.get(url) @auto_log
def post(self, url, files=None, data=None,headers=None):
if headers is None:
headers = self.headers
return self.session.post(url, headers=headers,files=files,data=data)

第二个类是用来操作QC的,现在我做了用户登录、搜索entity、添加附件三个功能。后续还会加上更新entity和创建entity等,把参考文档里的common tasks全都实现在这个类里。

class QCSession:
def __init__(self, username, password, qc_url):
self.api_client=RestClient()
auth = base64.b64encode("{}:{}".format(username, password))
self.api_client.session.headers['Authorization'] = "Basic {}".format(auth)
self.qc_url = qc_url
self.login() def login(self):
r = self.api_client.post(self.qc_url.auth, headers=self.api_client.session.headers)
assert(r.status_code is 200) def get(self, *args):
r = self.api_client.get(self.qc_url.path(*args))
assert (r.status_code is 200 or r.status_code is 201)
return r def add_attachment(self, filename, over_ride,*args):
data = {
'filename': ("", filename),
'override-existing-attachment': ("", over_ride)
}
files = {
'file': open(filename, 'rb')
}
r = self.api_client.post(self.qc_url.path(*args),data=data,files=files)
assert (r.status_code is 200 or r.status_code is 201)
return r def query_entitys(self, entity_name, query_expression, fields=None, page_size=""):
"""
:param entity_name: the entity to query
:param query_expression: the query expression
:param fields: the files to return in results
:param page_size: the result page size, by default is 2000 which is the max value
:return: resturn the BeautifulSoup parsed result
"""
query_url=u"%s?query={%s}&fields=%s&page-size=%s" %(entity_name,query_expression,fields,page_size)
r = self.get(query_url)
res = BeautifulSoup(r.content, "lxml")
return res

还有一个保存和生成QC的各种url地址的类:

class QCURL:
def __init__(self, hostname, domain, project, port=''):
self.__base = u'http://{}:{}/qcbin/'.format(hostname,port)
self.__work = self.__base + u'rest/domains/{}/projects/{}/'.format(domain,project)
self.auth = self.__base + u'authentication-point/authenticate'
self.logout = self.__base + u'authentication-point/logout/' def path(self, *args):
return self.__work + '/'.join([str(arg) for arg in args])

还有第四个类就是用QC Session类来实现我要的业务功能了。具体代码就不贴了,里面没什么可以让别人重用的东西。

拿上面三个类就可以用了。
给个例子:
登录QC,并查询指定的testset_id下所有的test-instances,并返回其id和test-id
下面所有没赋值的变量都换成自己环境的真实值的字符串就行了。

qc_url = QCURL(hostname, domain, project, port)
qc_session = QCSession(username, password, qc_url)
qc_session.query_entitys('test-instances', 'contains-test-set.id[%s]' % testset_id, "id,test-id")

希望这篇文章可以帮到一些有同样问题的人。以上代码在HP ALM 11.52上测试通过。

参考文档:http://alm-help.saas.hpe.com/en/12.53/api_refs/REST_TECH_PREVIEW/ALM_REST_API_TP.html

python小工具:用python操作HP的Quality Center的更多相关文章

  1. python小工具:用python操作HP的Quality Center (二)----- 用异步方式提高速度

    上接第一篇 http://www.cnblogs.com/sdet/p/6874631.html 在python中,很简单地能把http请求通过异步的方式发送,以下代码在python 3.6.0上运行 ...

  2. Python小工具--删除svn文件

    有的时候我们需要删除项目下的svn相关文件,但是SVN会在所有的目录下都创建隐藏文件.svn,手工一个个目录查找然后删除显然比较麻烦.所以这里提供了一个Python小工具用于批量删除svn的相关文件: ...

  3. python小工具myqr生成动态二维码

    python小工具myqr生成动态二维码 (一)安装 (二)使用 (一)安装 命令: pip install myqr 安装完成后,就可以在命令行中输入 myqr 查看下使用帮助: myqr --he ...

  4. 有哪些你不知道的python小工具

    python作为越来越流行的一种编程语言,不仅仅是因为它语言简单,有许多现成的包可以直接调用. python中还有大量的小工具,让你的python工作更有效率. 1.- 快速共享 - HTTP服务器 ...

  5. 分享一个 Linux 环境下,强力的Python 小工具

    场景 Linux 用户,经常需要在终端查看一些数据,从文件里看 或者网络协议获取数据并查看. 比如,查看文件里的json数据:比如,查看etcd里存下的数据. 如果直接看cat 或者 curl 得到的 ...

  6. 一个Json结构对比的Python小工具兼谈编程求解问题

    先上代码. jsondiff.py #!/usr/bin/python #_*_encoding:utf-8_*_ import argparse import json import sys rel ...

  7. 自制 Python小工具 将markdown文件转换成Html文件

    今天看到了一个Python库,名为markdown.瞬间就给了我一个灵感,那就是制作一个将markdown文件转换成html文件的小工具. 我的实验环境 操作系统: Windows 7 64位 旗舰版 ...

  8. Python小工具:利用ffmpy3库3秒钟将视频转换为音频

    作者 | pk 哥 来源公众号 | Python知识圈(ID:PythonCircle) 最近,有读者微信上私聊我,想让我写一篇视频批量转换成音频的文章,我答应了,周末宅家里把这个小工具做出来了. 这 ...

  9. 周报?谁还写周报啊?不都用Python小工具: 发个周报邮件给老板就行还写周报啊?不都用Python小工具: 发个周报邮件给老板就行

    缘起: 新跳槽到一家公司, 没想到第一个挑战居然是每周都要发周报. 告诉老板这周都干了些什么和下周准备干什么. 我记性不好, 常常忘事儿.所以很多时候周报都会忘记发送. 于是, 就决定写一个小工具好了 ...

随机推荐

  1. Oracle函数整理

    ) from dual;--绝对值 ,) from dual;--取模,取余数 select ceil (12.1) from dual;--去上限值 select floor (12.1) from ...

  2. 时效性福利:MindManager2017 破解攻略

    本文目的只是为了长期关注公众的活粉来谋个福利,24小时失效,没有提供盗版的意思 本文贡献的链接只存放2天,要下载的请从速~ 经过几个小时的奋斗,终于搞定了他,逆天我也终于可以从2016升级至2017~ ...

  3. webix .datatable 表格分页

    grid表格返回参数大都是 以下这种格式(参数名可能不一样) { data:[{...},{...} ...], count:39 } webix的参数格式为 { data:[{...},{...}, ...

  4. html/css/javascript的含义、作用及理解

    HTML(HyperText Markup Language/超文本标记语言) 含义:HTML是一种用于创建网页的标准标记语言. 作用:页面内可以包含图片.链接,甚至音乐.程序等非文字元素. 理解:主 ...

  5. Bar 3D 和Pie 3D的统计图形

    最近在做一个关于图形统计的界面,主要用到的是Dev控件ChartControl(功能很强大,能解决基本和复杂图形统计问题). ChartControl主要有Diagram.Series.Legend三 ...

  6. mac下CSV文件用FileReader、FileWriter读写乱码

      先说下windows的excel文件搬到mac下打开为什么会显示乱码.    在win下,excel采用GBK编码,1个汉字是存为2个字节,而mac下各种软件广泛默认使用UTF-8编码方式,如在e ...

  7. 发散问题——Spring容器及加载

    一.前言 发散问题系列,是围绕日常工作,发散思考,提取问题,并寻求答案的一个系列.总的来说,就是将遇到的问题发散来提出更多的问题,并通过解决发散问题,从而对问题有更深入的了解,对知识有更深刻的记忆,帮 ...

  8. 一次young gc耗时过长优化过程

    1    问题源起 上游系统通过公司rpc框架调用我们系统接口超时(默认超时时间为100ms)数量从50次/分突然上涨到2000次/分,在发生变化时间段里我们的系统也没有做过代码变更,但上游系统的调用 ...

  9. Spring BeanFactoryPostProcessor

    使用场景:当在配置文件中需要配置Bean(参数不同,class相同,id不同时)冗余的情况 继承 BeanFactoryPostProcessor 覆盖 postProcessBeanFactory( ...

  10. PROFINET如何实现实时性

    平时我们都听过文艺作品要“源于生活而高于生活”.PROFINET是基于工业以太网的,用文艺范儿的词汇说就是“源于以太网而高于以太网”.那么,PROFINET是怎么做到“高于以太网”的呢? 要做到比普通 ...