背景是这样的:

这个组的测试人员每跑一个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. webpack2.x基础属性讲解(一)

      webpack作为构建工具平时作为前端作为优化.模块编程.和分片打包的重要组成部分,大家可能并不陌生,如果没有时刻的去关注文档,那么大家可能不太清楚webpack已经默默然的升级到2.x了,对比1 ...

  2. GDOI2014模拟 旅行【SPFA】

    旅行(travel) 从前有一位旅者,他想要游遍天下所有的景点.这一天他来到了一个神奇的王国:在这片土地上,有n个城市,从1到n进行编号.王国中有m条道路,第i条道路连接着两个城市ai,bi,由于年代 ...

  3. 持续集成:TestNG组织如何测试用例 1

    持续集成:TestNG组织如何测试用例   poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.如果对课程感兴趣,请大家咨询qq:90 ...

  4. cooking构建工具报错MSBUILD :error MSB4132解决办法

    最近学习cooking构建工具的时候,在自己的笔记本上运行的好好的,项目在公司电脑上clone下来的时候,发现构建报错,逐条查错,试了好多方法也不行 最后在github上找到了答案,只是之前一直没找到 ...

  5. DataReader和DataSet区别

    可以使用DataReader类的对象或DataSet类的对象从数据库读取数据,但它们是有区别的,归纳起来大致有以下几条: 1.       DataReader是数据管理提供者类,而DataSet是一 ...

  6. 图解函数重载以及arguments

  7. PAT 1057

    Stack is one of the most fundamental data structures, which is based on the principle of Last In Fir ...

  8. Virtualbox 复制 CentOS 虚拟机无法联网

    Centos刚装好后无法联网 复制虚拟机后,出现 No such device eth0 我们要处理的三个问题: 在Virtualbox上安装好Centos后如何联网 如何在Virtualbox上复制 ...

  9. win7下用SSH连接linux虚拟机

    本文来自转载:原文 [需求] 在win7环境下用SSH(SecureShell)连接本地的一台虚拟机上ubuntu(11.10)系统  [环境] win7,ubuntu,vmware(8.0) [方案 ...

  10. JavaScript 小函数积累及性能优化

    获取值的类型: var toString = Object.prototype.toString; function getType(o) { return toString.call(o).slic ...