调用飞书接口导入供应商bug

news2025/1/11 5:53:09
1、业务背景

财务这边大部分系统都是供应商项目,由于供应商的研发人员没有飞书项目的权限,涉及到供应商系统需求 财务这边都是通过多维表格进行bug的生命周期管理如图:

但多维表格没有跟飞书项目直接关联,测试组做bug统计的时候无法计入供应商bug,对测试人员的bug数量造成一些影响。

解决方案:

读取表格数据,调用飞书项目接口,将bug导入到对应飞书项目需求里

2、结果展示
1、点击桌面“供应商bug导入”应用图标

2、输入文件路径和飞书项目ID

3、结果展示

导入结果:

原始数据:

备注:代码逻辑里对bug优先级做了映射,高级-- C类、中级-- B类、低级和建议-- C类

3、实现方式

官方文档: 飞书项目开发者手册

1、创建插件

插件入口

点击个人头像,从「开发者后台」进入插件开发者后台;

创建完成后,对插件进行数据权限管理和发布

插件凭证

插件凭证是插件开发阶段、运行阶段用于身份鉴权的唯一凭证。

2、名词解释

中文名词

API名词

描述

空间域名

simple_name

字段ID

field_key

工作项实例

instance

工作项实例是一个具体的事例,例如,一个已经创建的需求、缺陷、项目等

3、接口调用

通过基础名词解释可得,我们创建bug就是新增一个工作实例,找到官方文档中的创建工作项的接口信息

1、创建工作项接口

请求方式

POST

请求地址

/open_api/:project_key/work_item/create

请求header

详见 请求header

请求参数

参数类型

参数名

是否必填

值类型

说明

路径参数

project_key

string

空间id(project_key),或者空间域名(simple_name)

请求体参数

work_item_type_key

string

工作项类型,自定义工作项可通过获取空间下工作项类型获取

name

string

工作项名称

field_value_pairs

list<FieldValuePair>

FieldValuePair

创建工作项的具体字段可以从获取创建工作项元数据 接口中获取,字段格式可查看字段与属性解析格式

template_id

int64

模板ID,可以从以下途径获取:

  1. 获取创建工作项元数据接口中的template字段的options中获取,选择对应的value

  2. 获取工作项下的流程模板列表接口的template_id

  3. 获取字段信息接口中的template字段的options中获取,选择对应的value

请求体格式

​
{ "work_item_type_key": "story",//工作项类型 "template_id": 123123,//流程模板id "name":"2333",//工作项名称 "field_value_pairs": [ { "field_key": "description",//字段key,作为请求参数和field_alias二选一必填,选择范围为工作项元数据中获取的字段全集 "field_value": "2333"//字段值,作为请求参数必填;value填写规则取决于字段类型,不同类型的规则可见字段与属性解析格式  } ] }

​

返回格式

{ "data": 19781, // 工作项id "err": {}, "err_msg": "", "err_code": 0 }

根据接口文档描述,我们需要获得 请求header 信息。需要查看header文档

通过请求header文档,得知我们需要获取访问凭证

1、获取访问凭证接口

调用 获取插件访问凭证 接口,通过插件凭证 Plugin IDPlugin Secret获取 plugin_access_token (或者virtual_plugin_access_token

请求方式

POST

请求地址

https://{平台域名}/open_api/authen/plugin_token

请求头参数

参数名

类型

必填

说明

Content-Type

string

固定值:“application/json”

请求体参数

参数名

类型

必填

说明

plugin_id

string

插件唯一标识,Plugin ID

plugin_secret

string

插件密钥,Plugin Secret

type

int

插件访问凭证类型,可选值:0、1。默认为plugin_access_token,值为1时将返回virtual_plugin_access_token

cURL示例

curl --location 'https://{平台域名}/open_api/authen/plugin_token' \ --header 'Content-Type: application/json' \ --data '{ "plugin_id": "MII_63E9D49B8C820014", "plugin_secret": "D01B5F1A191C8620D133CDC371C0C7CB", "type": 0 }'

响应体参数

{ "data": { "expire_time": 7200, // token失效时间 "token": "p-49257489-f7d7-4cd6-b34f-98c6b81db375" // 插件访问凭证 plugin_access_token } }

"plugin_id" 和 "plugin_secret" 我们创建插件时已经获得,由接口文档,可以轻松获取到请求header的凭证信息

2、工作项类型获取

根据创建工作项的接口文档 work_item_type_key 需要通过 获取空间下工作项类型 接口获取

为了方便获取到我们需要的参数,只需要用postman获取即可

即: "work_item_type_key": "issue"

3、获取 template_id

同理我们可以调用获取字段信息接口,拿到缺陷的 template_id

即: "template_id": 34673,

4、其余自定义字段获取

可以通过获取工作项详情查询

最终的请求数据

{ "work_item_type_key": "issue", "name": data[i][0], "template_id": 34673, "field_value_pairs": [ { "field_alias": "bug_priority", "field_value": {"label": priority_map.get(data[i][6], {"label": "B类", "value": "53cnaxoz_"}).get("label"), "value": priority_map.get(data[i][6], {"label": "B类", "value": "53cnaxoz_"}).get("value") }, }, {"field_alias": "_field_linked_story", "field_value": linked_story}, {"field_alias": "owner", "field_value": reporter_map.get(data[i][1], "7230980664668045340")}, {"field_alias": "issue_reporter", "field_value": reporter_map.get(data[i][1], "7230980664668045340")}, {"field_alias": "issue_operator", "field_value": ["7413123527076806659"]}, {"field_alias": "description", "field_value": data[i][0]}, ] }

2、状态流转接口

状态流转接口新

请求方式

POST

请求地址

/open_api/:project_key/workflow/:work_item_type_key/:work_item_id/node/state_change

请求header

详见 请求header

请求参数

参数类型

参数名

是否必填

值类型

说明

路径参数

work_item_id

int64

工作项ID

work_item_type_key

string

工作项类型可以,从获取空间下工作项类型接口获取

project_key

string

空间id(project_key),或者空间域名(simple_name)

请求体参数

transition_id

int64

流转到下一状态的id,从获取工作流详情接口查询状态流获取

role_owners

list<RoleOwner>

RoleOwner

角色及负责人

fields

list<FieldValuePair>

FieldValuePair

要更新的字段数组(只能更新状态表单)

请求体格式

{ "transition_id":12345//状态id,仅状态流返回 "fields":[ { "field_alias": "sentry_link",//字段对接标识 "field_key": "field_658c22",//字段key "field_type_key": "text",//字段类型 "field_value": "23333"//字段值 } ] }

根据接口文档,我们需要获取header、work_item_id、work_item_type_key、project_key、transition_id等信息

其中header、work_item_type_key、project_key可以参考 “创建工作项” 接口里的获取方式

1、work_item_id的取值

work_item_id 我们可以从创建工作项接口的返回值里取到

2、transition_id 获取

transition_id 可以通过调用 获取工作流详情 接口获取

从接口返回信息可得,bug状态 :

由 OPEN --> RESOLVED 的 transition_id 是 983309

由 RESOLVED --> CLOSED 的 transition_id 是 983314

状态流转的请求json信息

{ "transition_id": 983309, # 状态改成 RESOLVED "fields": [ {"field_alias": "bug_remark", "field_value": "供应商bug"}, {"field_alias": "bug_reason", "field_type_key": "select", "field_key": "field_21fcfb", "field_value": {"label": "自测不仔细产生bug", "value": "b39yvbscm"}} ] }

4、代码示例
import pandas as pd
import json
import requests
from tkinter_test import imp_confirm

def get_plugin_token():
    """
    获取插件的 token并返回
    :return: 插件的token信息
    """
    url = 'https://project.feishu.cn/open_api/authen/plugin_token'
    payload = json.dumps({
        "plugin_id": plugin_id,
        "plugin_secret": plugin_secret,
        "type": 0
    })
    headers = {'Content-Type': 'application/json'}
    response = requests.request("POST", url, headers=headers, data=payload)
    return response.json()['data']['token']


def add_bug_datas(linked_story, file_path):
    """
    把bug导入到对应的飞书项目里
    :param linked_story:  飞书项目ID
    :param file_path: bug文件路径
    """
    df = pd.read_excel(file_path)
    data = df.iloc[0:, :]
    data = data.to_numpy()
    url = "https://project.feishu.cn/open_api/hdltech/work_item/create"
    for i in range(len(data)):
        user_data = {
            "work_item_type_key": "issue",
            "name": data[i][0],
            "template_id": 34673,
            "field_value_pairs": [
                {
                    "field_alias": "bug_priority",
                    "field_value": {"label": priority_map.get(data[i][6], {"label": "B类", "value": "53cnaxoz_"}).get("label"),
                                    "value": priority_map.get(data[i][6], {"label": "B类", "value": "53cnaxoz_"}).get("value")
                                    },
                },
                {"field_alias": "_field_linked_story", "field_value": linked_story},
                {"field_alias": "owner", "field_value": reporter_map.get(
                    data[i][1], "7230980664668045340")},
                {"field_alias": "issue_reporter", "field_value": reporter_map.get(
                    data[i][1], "7230980664668045340")},
                {"field_alias": "issue_operator",
                 "field_value": ["7413123527076806659"]},
                {"field_alias": "description", "field_value": data[i][0]},
            ]
        }
        data_to_send = json.dumps(user_data).encode("utf-8")
        header = {"content-type": "application/json",
                  "X-User-Key": user_key,
                  "X-PLUGIN-TOKEN": get_plugin_token()
                  }
        r = requests.post(url, data=data_to_send, headers=header)
        data_list.append(r.json()["data"])
        print(r.json())


def close_bug():
    """
    更改bug状态
    """
    headers = {
        "Content-Type": "application/json",
        "X-PLUGIN-TOKEN": get_plugin_token(),
        "X-USER-KEY": user_key,
    }
    for bug_id in data_list:
        bug_id = bug_id
        url = f"https://project.feishu.cn/open_api/hdltech/workflow/issue/{bug_id}/node/state_change"
        data = {
            "transition_id": 983309,  # 状态改成 RESOLVED
            "fields": [
                {"field_alias": "bug_remark", "field_value": "供应商bug"},
                {"field_alias": "bug_reason",
                 "field_type_key": "select",
                 "field_key": "field_21fcfb",
                 "field_value": {"label": "自测不仔细产生bug", "value": "b39yvbscm"}}
            ]
        }
        requests.post(url, json=data, headers=headers).json()
        data = {
            "transition_id": 983314,  # 状态改成close
            "fields": [
                {"field_alias": "bug_remark", "field_value": "供应商bug"},
            ]
        }
        result = requests.post(url, json=data, headers=headers).json()
        print(result)


file_path, linked_story = imp_confirm()
add_bug_datas(linked_story, file_path)
close_bug()
#print(file_path, linked_story)
4、弹窗信息的实现

Python GUI编程(Tkinter) :https://docs.python.org/zh-cn/3/library/tk.html

Tkinter文档

import tkinter as tk

root = tk.Tk()
root.geometry('400x230+500+260')
root.title('导入bug操作')
page = tk.Frame(root)
page.pack()

file_path = tk.StringVar()
story_id = tk.StringVar()

tk.Label(page).grid(row=0, column=0)
tk.Label(page, text='请输入文件路径').grid(row=1, column=1)
tk.Entry(page, textvariable=file_path).grid(row=1, column=2, pady=10)
tk.Label(page, text='请输入项目ID').grid(row=2, column=1, pady=10)
tk.Entry(page, textvariable=story_id).grid(row=2, column=2, pady=10)

def imp_confirm():
    filepath = file_path.get().replace(" ", "")
    storyid = story_id.get().replace(" ", "")
    if len(filepath) == 0:
        messagebox.showwarning("警告", "文件路径不能为空")
    elif len(storyid) == 0:
        messagebox.showwarning("警告", "项目ID不能为空")
    elif not storyid.isdigit():
        messagebox.showwarning('警告', '项目ID为整数请确认后输入')
    else:
        page.quit()
        #messagebox.showinfo("提示", "开始导入请稍后")
        return filepath, int(storyid)

tk.Button(page, text='确认', command=imp_confirm).grid(row=3, column=1, pady=10)
tk.Button(page, text='取消', command=page.quit).grid(row=3, column=2)

root.mainloop()
5、打包操作

使用 Pyinstaller 进行打包

pyinsatll中文文档

pyinstaller -w -i im.icns -n 供应商bug导入 opexcel.py

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

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

相关文章

【Docker】01-Docker常见指令

1. Docker Docker会下载镜像&#xff0c;运行的时候&#xff0c;创建一个隔离的环境&#xff0c;称为容器。 docker run -d \ # 创建并运行一个容器&#xff0c;-d表示后台运行 --name mysql \ # 容器名称-p 3307:3306 \ # 端口映射&#xff0c;宿主机端口映射到容器端口-e TZ…

打造同城O2O平台:外卖跑腿APP的架构与功能设计详解

今天&#xff0c;小编将于大家共同讨论外卖跑腿APP的架构设计及其核心功能&#xff0c;旨在为开发者提供一份详尽的参考。 一、外卖跑腿APP的架构设计 1.整体架构概述 通常包括前端、后端和数据库。 2.前端设计 用户端提供直观的界面&#xff0c;方便用户下单、查询订单状态…

文章解读与仿真程序复现思路——电力系统自动化EI\CSCD\北大核心《基于自适应时段划分的含氢微能网中长期变分辨率调度》

本专栏栏目提供文章与程序复现思路&#xff0c;具体已有的论文与论文源程序可翻阅本博主免费的专栏栏目《论文与完整程序》 论文与完整源程序_电网论文源程序的博客-CSDN博客https://blog.csdn.net/liang674027206/category_12531414.html 电网论文源程序-CSDN博客电网论文源…

Rust gRPC---Tonic实战

API 一个API做了两件事 客户端发起请求Request服务端作出响应Response REST是什么 REST&#xff08;Representational State Transfer&#xff09;&#xff1a;表现层状态传输&#xff0c;是一种设计风格&#xff0c;通常将 HTTP API 称为 RESTful API、RESTful 服务或 RES…

AI大模型生态暨算力大会今日举行,20位大咖领衔共探「AI NATIVE,生成未来」

出品&#xff5c;AI大模型工场 9月25日消息&#xff0c;由AI大模型工场主办AI大模型生态暨算力大会今日举行。作为国内最具影响力与最懂大模型的AI生态大会&#xff0c;大会讨论了AI大模型的最新进展和未来发展趋势。 2024年被业内称为大模型应用落地元年&#xff0c;大模型产…

【入门01】arcgis api 4.x 创建地图、添加图层、添加指北针、比例尺、图例、卷帘、图层控制、家控件(附完整源码)

1.效果 2.代码 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title></title><link rel"s…

JSP+Servlet+Mybatis实现列表显示和批量删除等功能

前言 使用JSP回显用户列表&#xff0c;可以进行批量删除&#xff08;有删除确认步骤&#xff09;&#xff0c;和修改用户数据&#xff08;用户数据回显步骤&#xff09;使用servlet处理传递进来的请求参数&#xff0c;并调用dao处理数据并返回使用mybatis&#xff0c;书写dao层…

828华为云征文|华为云Flexus云服务器X实例的网络性能测试

828华为云征文&#xff5c;华为云Flexus云服务器X实例的网络性能测试 前言一、Flexus云服务器X实例介绍1.1 Flexus云服务器X实例简介1.2 Flexus云服务器X实例特点1.3 Flexus云服务器X实例使用场景 二、iperf3工具介绍2.1 iperf3简介2.2 iperf3特点 三、本次实践介绍3.1 本次实践…

专利如何有效维持?

专利的获得并非一劳永逸&#xff0c;其法律效力的持续存在依赖于有效的专利维持工作。专利维持&#xff0c;即专利权人在专利有效期内&#xff0c;按照法定程序缴纳年费、提交必要的文件&#xff0c;以保持专利权的有效状态。这一过程对于确保创新成果持续闪耀、维护企业竞争优…

DSP——从入门到放弃系列——多核导航器(持续更新)

1、概述 C6678中的数据移动非常复杂&#xff0c;多核导航器是C6678中协助完成在器件内高速数据包移动的外围设备。 2、组成 多核导航器由1个队列管理子系统&#xff08;QMSS&#xff09;1个包DMA&#xff08;Packet DMA PKTDMA&#xff09;来控制和实现器件内的高速数据包移…

Arthas mc(Memory Compiler/内存编译器 )

文章目录 二、命令列表2.2 class/classloader相关命令2.2.2 mc &#xff08;Memory Compiler/内存编译器 &#xff09;举例1&#xff1a;可以通过-d命令指定输出目录&#xff1a;mc -d /temporary/tmp /temporary/tmp/AccountController.java举例2&#xff1a;通过--classLoade…

SpringBoot-全局处理异常,时间格式,跨域,拦截器,监听器

1.全局异常处理 使用ControllerAdvice与ExceptionHandler注解 /*** 全局异常处理程序** author * date */ ControllerAdvice ResponseBody public class GlobalExceptionHandler {ExceptionHandler(Exception.class)public JsonResult handleException(Exception e) {e.print…

Vue3 中 this 一分钟了解

Vue3 中 this 在Vue3的开发过程中&#xff0c;this的使用方式和Vue2有着显著的不同&#xff0c;特别是在组合式API&#xff08;Composition API&#xff09;的引入后。本文将深入探讨Vue3中this的使用&#xff0c;解析其底层源码&#xff0c;并探讨这种设计背后的原因&#xff…

Heart Animated Realistic 心脏运动模型素材带动画

Realistic Heart具有两个多边形质量的网格,具有详细的解剖结构,并配有高清纹理2048x2048,在高低多边形网格上具有高清法线贴图,可在教育、游戏和虚拟现实场景中获得更好、更真实的效果。 还具有完整的心动周期。 下载:​​Unity资源商店链接资源下载链接 效果图:

51单片机如何判断浮点数nan

使用这个函数进行判断 帮助信息内的描述如下 _chkfloat_ #include <intrins.h> unsigned char _chkfloat_ (float val); /* number for error checking */ Description: The _chkfloat_ function checks the status of a floating-point number. Return Value: The…

短视频去水印解析api接口使用文档

短视频去水印解析api接口&#xff0c;支持各大平台短视频和图集。 请求示例&#xff1a;https://www.dspqsy.vip/spapi?key密钥&url短视频链接 返回数据格式&#xff1a;JSON 请求方式&#xff1a;GET/POST 请求参数&#xff1a;url (短视频分享的URL) PHP 源码&…

C语言数组探秘:数据操控的艺术【下】

承接上篇&#xff0c;我们继续讲数组的内容。 八.二维数组的使用 当我们掌握了二维数组的创建和初始化&#xff0c;那我们怎么使用二维数组呢&#xff1f;其实二维数组访问也是使用下标的形式的&#xff0c;二维数组是有行和列的&#xff0c;只要锁定了行和列就能唯一锁定数组中…

Race Karts Pack 全管线 卡丁车赛车模型素材

是8辆高细节、可定制的赛车,内部有纹理。经过优化,可在手机游戏中使用。Unity车辆系统已实施-准备驾驶。 此套装包含8种不同的车辆,每种车辆有8-10种颜色变化,总共有75种车辆变化! 技术细节: -每辆卡丁车模型使用4种材料(车身、玻璃、车轮和BrakeFlare) 纹理大小: -车…

屏幕活动保存到NAS

目录 一、套件选择 二、员工机准备 1、下载安装ffmpeg 2、安装运行rtsp-simple-server 3、生成桌面流 4、接收查看桌面变化 三、NAS端配置 1、安装套件 2、配置Surveillence Station 3、实时监看 4、历史记录查看 5、录像文件操作 四、总结 朋友的朋友找上我,说到…

网络安全专业,在校大学生如何赚外快,实现财富自由?零基础入门到精通,收藏这一篇就够了

如今&#xff0c;计算机行业内卷严重&#xff0c;我们不找点赚外快的路子这么行呢&#xff1f; 今天就来说说网络安全专业平时都怎么赚外快。 一、安全众测 国内有很多成熟的src众测平台&#xff0c;如漏洞盒子、火线众测、补天、CNVD、漏洞银行等。一些大厂也有自己的src&a…