禅道Bug的一次迁移

一、场景

        平时工作记录在公司禅道上的问题想备份一份到本地,但是又没有公司禅道的数据库信息,有时候出测试报告想批量调整数据方便截图很困难,同时也为了学习禅道数据流转过程,所以有了把缺陷保存到本地一份的想法。

        实际上禅道支持导入缺陷,但是开源版不支持,所以就饶了一圈。

 二、思路:

        1.直接备份整个数据库行不通,考虑通过接口获取缺陷信息,生成sql插入到本地数据库中。

        2.不使用接口获取信息,直接导出缺陷数据,通过获取Excel信息生成sql插入到本地数据库。

这里使用思路一,因为禅道官方提供了二次开发指南,包括基本的接口文档等等,主要是更方便学习以接口的方式操作禅道。

三、操作流程

1.查看禅道官方文档,获取token是第一步

获取Token - 禅道二次开发手册 - 禅道项目管理软件

2.根据提供的文档,确定请求地址和方式,调试后可获取token。注:url_base(前置URL)可以根据登录禅道进行获取。例如:http://192.168.1.1:8080/zentao/user-login-L21heDQxMi8=.html,这个地址就是http://192.168.1.1:8080/zentao

3.获取bug信息。注:在官方手册上只能找到获取产品bug的接口文档,但是我的禅道上没有产品这个概念,只有项目。实际上把products改为projects可以直接获取项目的bug信息,其接口文档的评论区上也提供了其他方法,但是不好用。这里加上status、limit两个参数可以查看所有的bug。

4.信息可以拿到,开始分析数据库结构,主要是步骤中图片文件的关联关系,体现在zt_file和zt_bug两张表。

        对应关系1:zt_bug表中step字段文件名的前缀数字即对应zt_file文件的id

        对应关系2:zt_file表文件pathname和图片路径的关联关系,比如2024/12/1.png对应的就是ZenTao/app/zentao/www/data/upload/1/202412/1.png。注:因为是插入的数据,所以pathname就用的id,实际上禅道会对创建产生的图片文件会进行加密,肯定不会叫这个名字

5.理清接口调用和数据库关联关系后,开始写代码。这里使用python进行操作。代码结构如下:其中获取sql等场景操作都在scene下,其余都是封装的一些方法。

5.1 获取sql所需要的数据信息

from api.zentao import zentao
import json
from common.json_extractor import extractor
from common.read_data import data
class Get_sqlData():def __init__(self):self.data = Nonedef get_sql_data(self,header,json_data):json_data = json.dumps(json_data)res = zentao.zentao_login(data=json_data, headers=header)data.update_ini("../config/setting.ini", "token", "token", res.json()['token'])print(res.json()['token'])header = {"Content-Type": "application/json","token": res.json()['token']}res = zentao.zentao_bug(id="5", headers=header)# print(res.json())title_list = extractor.extract_by_jsonpath(res.json(), "$.bugs[*].title")steps_list = extractor.extract_by_jsonpath(res.json(), "$.bugs[*].steps")# print(steps_list)pattern = r'\{(\d+)\.(?:jpeg|png)\}'extracted_numbers = extractor.extract_numbers_from_img_tags(steps_list, pattern)# print(extracted_numbers)return title_list,steps_list,extracted_numbers
get_sqlData = Get_sqlData()
if __name__ == '__main__':header = {"Content-Type": "application/json"}json_data = {"account" : "admin","password" : "123456"}res = get_sqlData.get_sql_data(header,json_data)print(res)

5.2 生成sql语句,main函数中注释的部分用于切换生成哪个sql语句,这里需要生成zt_file和zt_bug两个表的sql语句。

from get_sql_data import get_sqlDataclass GenerateSql():def __init__(self):self.data = Nonedef generate_sql_bug(self,sql_template,title_list=None,step_list=None,file_id_list=None):if file_id_list==None:sql_statements = []for title,step in zip(title_list,step_list):# 使用str.format()方法替换占位符sql_statement = sql_template.format(title=title.replace("'", "''"),  # 转义单引号以防止SQL语法错误steps=step.replace("'", "''")  # 同样转义HTML标签中的单引号)sql_statements.append(sql_statement)return sql_statementselse:sql_statements = []for id in file_id_list:sql_statement = sql_template.format(id=id.replace("'", "''"),  # 转义单引号以防止SQL语法错误)sql_statements.append(sql_statement)return sql_statementsgenerate_sql = GenerateSql()
if __name__ == '__main__':sql_template = """INSERT INTO `zentao`.`zt_bug` (`product`, `title`, `severity`, `pri`, `type`,`steps`, `status`, `openedBy`, `openedDate`, `openedBuild`,`assignedTo`, `assignedDate`, `resolvedBy`, `resolution`,`resolvedBuild`, `resolvedDate`, `closedBy`, `closedDate`,`lastEditedBy`, `lastEditedDate`) VALUES (1, '{title}', 3, 3, 'codeerror','{steps}','closed', 'admin', '2024-12-09 17:06:17', 'trunk','admin', '2024-12-10 16:33:15','admin', 'fixed', 'trunk', '2024-12-10 00:00:00','admin', '2024-12-10 00:00:00','admin', '2024-12-10 16:33:15');"""sql_template_file = """INSERT INTO `zentao`.`zt_file` ( `id`, `pathname`, `title`, `extension`, `objectType`, `objectID`, `addedBy`, `addedDate`, `downloads`, `extra`, `deleted` )VALUES( {id}, '202412/{id}.png', 'image.png', 'png', '', 0, 'admin', '2024-12-10 00:00:00', 0, '', '0' );"""
#    print(type(sql))header = {"Content-Type": "application/json"}data = {"account" : "admin","password" : "123456"}data_list = get_sqlData.get_sql_data(header,data)# sql = generate_sql.generate_sql_bug(sql_template,title_list=data_list[0],step_list=data_list[1])# for i in sql:#     print(i)sql =generate_sql.generate_sql_bug(sql_template_file,file_id_list=data_list[2])for i in sql:print(i)

5.3 下载图片至本地禅道路径

from concurrent.futures import ThreadPoolExecutor, as_completed
from common.read_data import data
from get_sql_data import get_sqlData
import os
import requests
from urllib.parse import urlparse, unquote
from common.logger import logger
from PIL import Image
import iodef download_image(url, save_path, id):# 确保目录存在if not os.path.exists(save_path):os.makedirs(save_path)try:token = data.load_ini("../config/setting.ini")['token']['token']header = {"token": token}response = requests.get(url, headers=header, stream=True)if response.status_code == 200:try:img = Image.open(io.BytesIO(response.content))filename = f"{id}.png"img = img.convert("RGB")  # 确保图片模式兼容JPEGfile_path = os.path.join(save_path, filename)img.save(file_path, 'JPEG')logger.info(f"Downloaded and saved as JPEG: {url} to {filename}")except Exception as e:logger.error(f"Error processing image from {url}: {e}")else:logger.error(f"Failed to download image {url}")with open(file_path, 'wb') as file:file.write(response.content)print(f"Downloaded: {filename}")except Exception as e:print(f"Failed to download {url}. Error: {e}")def download_images(url_list, save_path, id_list, max_workers=5):# 使用线程池并发下载图片with ThreadPoolExecutor(max_workers=max_workers) as executor:futures = [executor.submit(download_image, url, save_path, id) for url, id in zip(url_list, id_list)]for future in as_completed(futures):future.result()  # 获取结果并处理异常(如果有的话)if __name__ == '__main__':header = {"Content-Type": "application/json"}json_data = {"account": "admin","password": "123456"}res = get_sqlData.get_sql_data(header, json_data)id = res[2]url_base = data.load_ini("../config/setting.ini")['host_Portal_System_Test']['api_root_url']url_list = [url_base + f"/file-read-{i}.png" for i in id]print(url_list)path = r"D:\ZenTao\app\zentao\www\data\upload\1\202412"# 调用函数下载图片download_images(url_list, path, id_list=id)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/63986.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Redis - 消息队列 Stream

一、概述 消息队列 定义 消息队列模型:一种分布式系统中的消息传递方案,由消息队列、生产者和消费者组成消息队列:负责存储和管理消息的中间件,也称为消息代理(Message Broker)生产者:负责 产…

C语言数组和字符串笔记

C语言数组和字符串笔记 1. 数组及其相关概念 1.1 为什么需要使用数组? 数组是一个有序的、类型相同的数据集合。这些数据被称为数组的元素。每个数组都有一个名字,数组名代表数组的起始地址。数组的元素通过索引或下标访问,索引从0开始。 …

双目摄像头标定方法

打开matlab 找到这个标定 将双目左右目拍的图像上传(左右目最好不少于20张) 等待即可 此时已经完成标定,左下角为反投影误差,右边为外参可视化 把这些误差大的删除即可。 点击导出 此时回到主页面,即可看到成功导出 Ca…

数据结构开始——时间复杂度和空间复杂度知识点笔记总结

好了,经过了漫长的时间学习c语言语法知识,现在我们到了数据结构的学习。 首先,我们得思考一下 什么是数据结构? 数据结构(Data Structure)是计算机存储、组织数据的方式,指相互之间存在一种或多种特定关系的数据元素…

什么是MMD Maximum Mean Discrepancy 最大均值差异?

9多次在迁移学习看到了,居然还是Bernhard Schlkopf大佬的论文,仔细看看。 一.什么是MMD? 1. MMD要做什么? 判断两个样本(族)是不是来自于同一分布 2.怎么做?(直观上)…

电梯内电动车识别数据集,可准确识别电梯内是否有电动车 支持YOLO,COCO,VOC三种格式的标注 7111张图片

电梯内电动车识别数据集,可识别电梯内是否有电动车 支持YOLO,COCO,VOC三种格式的标注 7111张图片 7111总图像数 数据集分割 训练组 74% 5291图片 有效集 16% 1168图片 测试集 9% 652…

Collection接口

目录 一. Collection基本介绍 二. Collection中的方法及其使用 1. 添加元素 (1) 添加单个元素 (2) 添加另一集合中的所有元素 2. 删除元素 (1) 删除单个元素 (2) 删除某个集合中包含在其他集合中的元素 (3) 保留两个集合中的交集部分, 删除其他元素. 3. 遍历元素 (1) …

Mybatis Plus 3.0 快速入门

1、简介 MyBatis-Plus (简称 MP)是一个 MyBatis 的增强工具,在 MyBatis 的基础上只做增强不做改变,为简化开发、提高效率而生。 2、创建并初始化数据库 2.1、创建数据库 mybatis_plus 2.2、创建 User 表 其表结构如下: idnameageemail1Jone18test1@baomidou.com2Jack…

Verilog实现图像处理的行缓存Line Buffer

在图像处理中,难免会遇到对图像进行卷积或者模板的局部处理,例如ISP中的一些算法,很大部分都需要一个窗口,在实时视频处理中,可以利用行缓存Line buffer可以暂存几行数据,然后同时输出每行中的对应列的像素…

【银河麒麟高级服务器操作系统】有关dd及cp测试差异的现象分析详解

了解更多银河麒麟操作系统全新产品,请点击访问 麒麟软件产品专区:https://product.kylinos.cn 开发者专区:https://developer.kylinos.cn 文档中心:https://documentkylinos.cn dd现象 使用银河麒麟高级服务器操作系统执行两次…

ORACLE逗号分隔的字符串字段,关联表查询

使用场景如下: oracle12 以前的写法: selectt.pro_ids,wm_concat(t1.name) pro_names from info t,product t1 where instr(,||t.pro_ids|| ,,,|| t1.id|| ,) > 0 group by pro_ids oracle12 以后的写法: selectt.pro_ids,listagg(DIS…

记录2024-leetcode-字符串DP

10. 正则表达式匹配 - 力扣(LeetCode)

微信开发者工具(小程序)的版本管理,Git Push 和 Pull

微信开发者工具(小程序)的版本管理,Git Push 和 Pull 一、设置 第一次用微信开发者工具自带的版本管理的拉取和推送功能,稍稍的研究了下。 1、首先要先设置 “用户”,名字和邮箱,不一定要真名&#xff0c…

2020-12-07 光棍数

由光棍数的特征可推导其商的个位数不存在偶数且只有1、3、7、9这4个数。一个数可匹配多个光棍数且必定是中间隔着0的循环数。 void 光棍数(int n) {//缘由http://ask.csdn.net/questions/3444069 做乘法运行时间超长int w 0; long long x 111111111111111, j 0;//j x*n;/…

【Linux系统】—— 初识 shell 与 Linux 中的用户

【Linux系统】—— 初识shell 与 Linux 中的用户 1 Xshell 运行原理1.1 命令行的组成1.2 外壳程序 2 Linux中的用户2.1 两种用户2.2 创建普通用户2.3 用户切换2.3.1 普通->超级2.3.2 超级->普通 3 指令的短暂提权3.1 为什么要提权3.2 sudo 指令3.3 人人都能提权吗 1 Xshe…

.NET平台使用C#设置Excel单元格数值格式

设置Excel单元格的数字格式是创建、修改和格式化Excel文档的关键步骤之一,它不仅确保了数据的正确表示,还能够增强数据的可读性和专业性。正确的数字格式可以帮助用户更直观地理解数值的意义,减少误解,并且对于自动化报告生成、财…

Android显示系统(10)- SurfaceFlinger内部结构

一、前言: 之前讲述了native层如何使用SurfaceFlinger,我们只是看到了简单的API调用,从本文开始,我们逐步进行SurfaceFlinger内部结构的分析。话不多说,莱茨狗~ 二、类图: 2.1、总体架构: 先看下SurfaceFlinger的关键成员和我们BootAnimation侧关键成员如何对应起来…

深度学习中的多通道卷积与偏置过程详解

目录 ​编辑 多通道卷积的深入理解 🔍 卷积核的多维特性 🌌 卷积操作的细节 🔧 多通道卷积的优势 🌟 偏置过程的深入理解 🎯 偏置的两种实现方式 🛠️ 偏置的作用与重要性 🌈 多通道卷…

易语言鼠标轨迹算法(游戏防检测算法)

一.简介 鼠标轨迹算法是一种模拟人类鼠标操作的程序,它能够模拟出自然而真实的鼠标移动路径。 鼠标轨迹算法的底层实现采用C/C语言,原因在于C/C提供了高性能的执行能力和直接访问操作系统底层资源的能力。 鼠标轨迹算法具有以下优势: 模拟…

【蓝桥杯选拔赛真题93】Scratch青蛙过河 第十五届蓝桥杯scratch图形化编程 少儿编程创意编程选拔赛真题解析

目录 Scratch青蛙过河 一、题目要求 编程实现 二、案例分析 1、角色分析 2、背景分析 3、前期准备 三、解题思路 1、思路分析 2、详细过程 四、程序编写 五、考点分析 六、推荐资料 1、入门基础 2、蓝桥杯比赛 3、考级资料 4、视频课程 5、python资料 Scratc…