实际工作中通过python+go-cqhttp+selenium实现自动检测维护升级并发送QQ通知消息(程序内测)

news2025/1/11 2:02:40

说明:该篇博客是博主一字一码编写的,实属不易,请尊重原创,谢谢大家!

首先,今年比较忙没有多余时间去实操创作分享文章给大家,那就给大家分享下博主在实际工作中的一点点内容吧,就当交流交流~

需求

叙述

目前公司有个跨平台大项目正在内测中,是基于QT框架研发的客户端应用程序

客户端程序的更新不像web端程序只需要清理缓存(存在js更新时)刷新即可更新至最新代码,就需要服务端维护升级批次->客户端检测更新->拉取升级列表下载批次文件->替换程序目录下的文件(数据库增量升级以及脚本文件)

当程序代码打包至公司内网升级目录下,每次都需要去通知维护人,维护人则需要在升级平台维护及开放程序版本批次,整个流程如下:

1、登录进入升级平台
2、选择项目
2.1 Windows64
2.2 Windows32
2.3 统信aarch64
2.4 统信amd64
2.5 统信arm64
2.6 银河麒麟loongarch64
2.7 银河麒麟arm64
2.8 银河麒麟amd64
2.9 中标麒麟arm64
2.10 中标麒麟amd64
2.11 MacX86_64
2.12 MacArm64
3、新建批次
4、导入本地文件
5、上传到下载服务器
6、升级说明
7、产品版本号和用户显示版本号配置
8、开放批次
8.1 内测开放
8.2 正式开放

PS:由于是公司还未上线的项目,所以不能细致透露

痛点1:每次需要研发经理通知(存在忘记通知或延迟通知)

痛点2:手动维护繁琐枯燥,批次版本信息容易维护错误

痛点3:忘记或延迟通知相应人员进行测试

解决

叙述

与研发经理进行约定,每次程序打包生成到指定的共享目录,编写程序进行10s检测目录下是否有批次版本升级文件产生,如果有则进行记录并自动化进行维护升级批次,开放批次后并下发通知消息到指定QQ群

解决痛点1:

通过last_state.cfg 配置文件存储上一次(或第一次)目录的状态,每个目录记录其最后一次修改时间,启动项目或项目运行期间以此时间进行判断是否目录有更新

# 要检查的目录列表
directories_to_check = [r"N:\windows\内测\32", r"N:\windows\内测\64", r"N:\windows\公测\32", r"N:\windows\公测\64", r"N:\uos\公测\aarch64"]

last_state = {}  # 上一次的目录状态字典

upgrade_flag = None

# 从配置中读取上一次的目录状态
def read_last_state():
    global upgrade_flag
    if os.path.exists('last_state.cfg'):
        with open('last_state.cfg', 'r') as f:
            for line in f:
                path, last_time = line.strip().split('|')
                last_state[path] = float(last_time)
        upgrade_flag = True
    else:
        upgrade_flag = False


# 更新上一次的目录状态到配置
def update_last_state():
    global upgrade_flag
    upgrade_flag = True
    with open('last_state.cfg', 'w') as f:
        for path, last_time in last_state.items():
            f.write(f'{path}|{last_time}\n')


# 循环判断多个目录下是否有新文件产生,并输出文件名和目录
def watch_dirs():
    while True:
        for directory_path in directories_to_check:
            for filename in os.listdir(directory_path):
                path = os.path.join(directory_path, filename)
                mtime = os.path.getmtime(path)
                if path not in last_state:
                    new_directory_flag = True
                else:
                    cfg_mtime = float(last_state[path])
                    if mtime > cfg_mtime:
                        new_directory_flag = True
                    else:
                        new_directory_flag = False
                if new_directory_flag:
                    n_time = datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S")
                    n_filename = "{%s}" % filename
                    f_path = r"{}\{}".format(directory_path, filename)
                    if upgrade_flag:
                        print(f'{n_time} 发现新目录在 {directory_path} 中:{n_filename}')
                        # 增量升级复制文件
                        cp_file(f_path, filename)
                        # 调用升级
                        auto_update(f_path, filename)
                    # 升级完成后获取最新的目录修改时间并赋值给对应path
                    mtime = os.path.getmtime(path)
                    last_state[path] = mtime  # 更新目录状态
        update_last_state()  # 更新目录状态到配置
        time.sleep(10)  # 每10秒检查一次

在这里插入图片描述

解决痛点2:

登录升级平台,1选择项目——2新增升级批次——3导入文件——4上传至下载服务器——5版本配置——6升级说明——7开放批次,使用selenium框架进行自动化处理

在这里插入图片描述

def auto_update(f_path, batch):
    len_file = len(os.listdir(f_path))
    if len_file > 0:
        new_f_path = f_path.replace("\\", "_")
        file_path_list = new_f_path.split("_")
        os_name = file_path_list[1]
        env = file_path_list[2]
        frame_num = file_path_list[3]
        upgrade_str = random.choice(str_lst)
        if os_name == 'uos':
            os_name = "统信"
        os_type = os_name+frame_num
        # print(os_type)
        options = webdriver.EdgeOptions()
        options.add_experimental_option('detach', True)  # 不自动关闭浏览器
        driver = webdriver.Edge(options=options)  # 引入edge驱动
        driver.maximize_window()
        driver.get("http://xxxx")
        # 登录平台
        driver.find_element(by=By.XPATH, value='//*[@id="input_box"]/input').send_keys('yourname')
        driver.find_element(by=By.XPATH, value='//*[@id="login_box"]/div[2]/input').send_keys('yourpwd')
        driver.find_element(by=By.XPATH, value='//*[@id="login_box"]/button').click()
        time.sleep(1)
        # todo:切换项目名称
        driver.find_element(by=By.XPATH, value='//*[@id="u20"]').click()
        time.sleep(1)
        ul = driver.find_element(by=By.XPATH, value='/html/body/div[5]/div/div/div/ul')
        all_project_list = ul.find_elements(by=By.XPATH, value='li')
        # print(all_project_list, type(all_project_list))  # 计算有多少个li
        index = 0
        pro_num = len(all_project_list)
        for p_name in all_project_list:
            index+=1
            # print(index, p_name.text)
            if os_type in p_name.text.lower():
                x_path = '/html/body/div[5]/div/div/div/ul/li[{}]/span/a'.format(index)
                driver.find_element(by=By.XPATH, value=x_path).click()
        time.sleep(1)
        # list[-1].text  # 用列表标识符取最后一个li
        # todo:新增升级批次
        driver.find_element(by=By.CSS_SELECTOR, value='#u17_div > div > span > svg').click()
        time.sleep(0.5)
        y = batch.split(".")[2][:2]
        m = batch.split(".")[2][2:]
        u_m = batch.split(".")[2][2:].replace("0", "")
        d = int(batch.split(".")[-1][:2]) - 10
        if len(str(d)) == 1:
            u_d = "0{}".format(d)
        else:
            u_d = d
        new_batch_day = "20{}.{}.{}".format(y, m, u_d)
        u_new_batch_day = "20{}.{}.{}".format(y, u_m, d)
        # print(batch, to_day, new_batch_day, m)
        u_batch = batch[-3:]
        if to_day != new_batch_day:
            b_xpath = '/html/body/div[last()]/div/div[2]/div/div[2]/div[2]/div/input'
            driver.find_element(by=By.XPATH, value=b_xpath).clear()
            time.sleep(0.5)
            driver.find_element(by=By.XPATH, value=b_xpath).send_keys(u_new_batch_day)
            u_batch_day = "{}.{}".format(new_batch_day, u_batch)
        else:
            u_batch_day = "{}.{}".format(to_day, u_batch)
        n_xpath = '/html/body/div[last()]/div/div[2]/div/div[2]/div[3]/button[2]/span'
        driver.find_element(by=By.XPATH, value=n_xpath).click()
        time.sleep(1)
        # todo:切换升级说明
        driver.find_element(by=By.XPATH, value='//*[@id="u27_div"]/ul/li[3]/span/div/span').click()
        time.sleep(0.5)
        driver.find_element(by=By.XPATH, value='//*[@id="u27_div"]/div[3]/textarea').send_keys(upgrade_str)
        time.sleep(0.5)
        # todo:切换版本配置
        driver.find_element(by=By.XPATH, value='//*[@id="u27_div"]/ul/li[2]/span/div/span').click()
        time.sleep(0.5)
        driver.find_element(by=By.XPATH, value='//*[@id="u27_div"]/div[2]/div[1]/input').send_keys(batch)
        time.sleep(0.5)
        driver.find_element(by=By.XPATH, value='//*[@id="u27_div"]/div[2]/div[2]/input').send_keys(u_batch_day)
        time.sleep(0.5)
        # todo:切换文件配置
        driver.find_element(by=By.XPATH, value='//*[@id="u27_div"]/ul/li[1]/span/div/span').click()
        # todo 导入文件
        driver.find_element(by=By.XPATH, value='//*[@id="u27_div"]/div[1]/div/div[1]/button[1]/span[2]').click()
        time.sleep(0.5)
        dr_xpath = '/html/body/div[last()]/div/div[2]/div/div[2]/div[2]/div/span/div[1]/span/div/button/span[2]'
        driver.find_element(by=By.XPATH, value=dr_xpath).click()
        time.sleep(1)
        # todo 上传文件
        # 按shift以及松shift键
        win32api.keybd_event(16, 0, 0, 0)
        win32api.keybd_event(16, 0, win32con.KEYEVENTF_KEYUP, 0)
        time.sleep(0.5)
        autoit.send(f_path)
        time.sleep(1)
        win32api.keybd_event(13, 0, 0, 0)
        win32api.keybd_event(13, 0, win32con.KEYEVENTF_KEYUP, 0)
        win32api.keybd_event(13, 0, 0, 0)
        win32api.keybd_event(13, 0, win32con.KEYEVENTF_KEYUP, 0)
        time.sleep(1)
        # todo 计算打开弹窗居中的坐标,并移动至此
        c_x, c_y = autoit.win_get_client_size("打开")
        m_x = int(int(center_x) - (int(c_x) / 2))
        m_y = int(int(center_y) - (int(c_y) / 2))
        autoit.win_move("打开", m_x, m_y)
        time.sleep(1)
        autoit.mouse_click("left", int(center_x), int(center_y))
        time.sleep(0.5)
        win32api.keybd_event(17, 0, 0, 0)
        win32api.keybd_event(65, 0, 0, 0)
        win32api.keybd_event(65, 0, win32con.KEYEVENTF_KEYUP, 0)
        win32api.keybd_event(17, 0, win32con.KEYEVENTF_KEYUP, 0)
        time.sleep(0.5)
        win32api.keybd_event(13, 0, 0, 0)
        win32api.keybd_event(13, 0, win32con.KEYEVENTF_KEYUP, 0)
        time.sleep(1)
        qr_xpath = "/html/body/div[last()]/div/div[2]/div/div[2]/div[3]/button[2]/span"
        driver.find_element(by=By.XPATH, value=qr_xpath).click()
        len_file = len(os.listdir(f_path))
        print("{}目录文件个数为:{}个".format(batch, len_file))
        if 0 < len_file <= 20:
            print("开始导入文件,等待10秒....")
            time.sleep(10)
        else:
            print("开始导入文件,等待30秒....")
            time.sleep(30)
        driver.find_element(by=By.XPATH, value='//*[@id="table"]/div/div[1]/table/thead/tr[1]/th[1]/div/label/span/span').click()
        time.sleep(0.5)
        driver.find_element(by=By.XPATH, value='//*[@id="u27_div"]/div[1]/div/div[1]/button[2]/span[2]').click()
        time.sleep(0.5)
        driver.find_element(by=By.XPATH, value='//*[@id="tablefwq"]/div/div[1]/table/thead/tr[1]/th[1]/label/span/span').click()
        time.sleep(0.5)
        driver.find_element(by=By.CSS_SELECTOR, value='body > div:last-child > div > div.ant-modal-wrap > div > div.ant-modal-content > div.ant-modal-footer > button.ant-btn.ant-btn-primary > span').click()
        if 0 < len_file <= 20:
            print("开始上传文件至下载服务器,等待15秒....")
            time.sleep(15)
        else:
            print("开始上传文件至下载服务器,等待45秒....")
            time.sleep(45)
        # todo:下拉滚动底部,开放版本
        driver.find_element(by=By.XPATH, value='//*[@id="u26_div"]/div[2]').click()
        time.sleep(0.2)
        win32api.keybd_event(35, 0, 0, 0)
        win32api.keybd_event(35, 0, win32con.KEYEVENTF_KEYUP, 0)
        time.sleep(0.2)
        driver.find_element(by=By.XPATH, value='//*/li[last()]/span/div/div/div/span[1]').click()
        time.sleep(0.5)
        driver.switch_to.default_content()
        time.sleep(0.5)
        driver.quit()
    else:
        print("警告!目录:{} 下的升级文件为空,将不进行升级调用!".format(f_path))

解决痛点3:

维护升级批次成功后,将向指定QQ群发送自定义消息,这里需要借助go-cqhttp框架,下载解压后,按一下步骤进行配置即可

  • Step1:下载后解压go-cqhttp_windows_amd64.zip,点击运行exe

    在这里插入图片描述
  • Step2:运行成功后,会生成go-cqhttp.bat文件,再运行这个批处理文件,出现如下窗口
    在这里插入图片描述
  • Step3:选择0,回车;编辑生成配置文件config.yml,切记只填写qq号(密码不填写,选择扫码登录,这样更安全且不会出现错误)
    在这里插入图片描述
  • Step4:编辑打开目录下产生的device.json文件,修改其中protocol的值为2,否则一直登陆失败
    在这里插入图片描述

配置完成后,拿出你的手机打开QQ进行扫码登录,登录后控制台日志出现警告不用管

在这里插入图片描述

接下来就是编写一个def方法来实现与go-cqhttp框架的交互,其实原理就是监听本地5700 socket消息,就跟以前飞书发消息回复消息是一样原理

def send_msg(resp_dict):
    client = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    ip = '127.0.0.1'
    client.connect((ip, 5700))
    msg_type = resp_dict['msg_type']  # 回复类型(群聊/私聊)
    number = resp_dict['number']  # 回复账号(群号/好友号)
    msg = resp_dict['msg']  # 要回复的消息
    # 将字符中的特殊字符进行url编码
    msg = msg.replace(" ", "%20")
    msg = msg.replace("\n", "%0a")
    if msg_type == 'group':
        payload = "GET /send_group_msg?group_id=" + number + "&message=" + msg + " HTTP/1.1\r\nHost:" + ip + ":5700\r\nConnection: close\r\n\r\n"
    elif msg_type == 'private':
        payload = "GET /send_private_msg?user_id=" + number + "&message=" + msg + " HTTP/1.1\r\nHost:" + ip + ":5700\r\nConnection: close\r\n\r\n"
    new_time = datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S")
    print("{} 发送{}".format(new_time, payload))
    client.send(payload.encode("utf-8"))
    client.close()

最后在auto_update方法中调用send_msg方法即可

if env == "内测":
    driver.find_element(by=By.CSS_SELECTOR, value='body > div:nth-last-child(2) > div > div > div > ul > li:nth-child(2) > span > a').click()
    msg = "内测升级 {}_{}_{}".format(os_name, frame_num, batch)
else:
    driver.find_element(by=By.CSS_SELECTOR, value='body > div:nth-last-child(2) > div > div > div > ul > li:nth-child(3) > span > a').click()
    msg = "正式升级 {}_{}_{}".format(os_name, frame_num, batch)
resp_group_dict = {'msg_type': 'group', 'number': 'QQ群号', 'msg': msg}
send_msg(resp_group_dict)

效果截图:

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

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

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

相关文章

2024考研408-计算机网络 第四章-网络层学习笔记

文章目录 前言一、网络层的功能1.1、网络层功能概述&#xff08;三种功能介绍&#xff09;1.2、SDN基本概念1.2.1、理解转发与路由选择1.2.1.1、转发1.2.1.2、路由选择 1.2.2、数据平面&#xff08;转发&#xff09;1.2.3、控制平面&#xff08;路由计算与选择&#xff09;实现…

九分学长大作文笔记

前言 最近看了《顾家北手把手教你雅思写作》&#xff0c;我觉得收获很少&#xff0c;逻辑感觉很乱&#xff0c;不知道为什么网上全是吹的。在Bilibili上看了九分学长&#xff0c;非常清晰&#xff0c;发现自己很多没有注意到的问题&#xff0c;下面将总结一些对我有用的部分。…

C#,OpenCV开发指南(01)

C#&#xff0c;OpenCV开发指南&#xff08;01&#xff09; 一、OpenCV的安装1、需要安装两个拓展包&#xff1a;OpenCvSharp4和OpenCvSharp4.runtime.win 二、C#使用OpenCV的一些代码1、需要加头文件2、读取图片3、在图片上画矩形框4、 在图片上画直线 一、OpenCV的安装 1、需…

使用 React Native CLI 创建项目

React Native 安装的先决条件和设置 需要掌握的知识点 掌握 JavaScript 基础知识掌握 React 相关基础知识掌握 TypeScript 相关基础知识 安装软件前需要首先安装Chocolatey。Chocolatey 是一种流行的 Windows 包管理器。 安装 nodejs 和 JDK choco install -y nodejs-lts …

接口测试——postman接口测试(三)

目录 1. postman介绍与安装 2. postman发送get请求 3. postman发送post请求 1. postman介绍与安装 安装网址&#xff1a;Postman安装教程&#xff1a;留言找我要即可 2. postman发送get请求 import pymysql from flask import Flask,request# 这里是mysql的基本连接信息 c…

cloudstack management高可用

一、环境说明 CPU&#xff1a; kunpeng 920 操作系统&#xff1a;OpenEuler 22.03 IP角色192.168.157.20mysql192.168.157.21management-server 1192.168.157.22management-server 2192.168.157.30nginx 二、部署 基础环境准备参考【cloudstack测试环境搭建】 1、部署mysql&…

音视频 FFmpeg命令行搭建

文章目录 一、配置二、测试 一、配置 以FFmpeg4.2.1 win32为例 解压ffmpeg-4.2.1-win32-shared.zip 拷⻉可执⾏⽂件到C:\Windows拷⻉动态链接库到C:\Windows\SysWOW64 注&#xff1a;WoW64 (Windows On Windows64)是⼀个Windows操作系统的⼦系统&#xff0c;被设计⽤来处理许…

替换开源LDAP,某科技企业用宁盾目录统一身份,为业务敏捷提供支撑

客户介绍 某高科技企业成立于2015年&#xff0c;是一家深耕于大物流领域的人工智能公司&#xff0c;迄今为止已为全球16个国家和地区&#xff0c;120余家客户打造智能化升级体验&#xff0c;场景覆盖海陆空铁、工厂等货运物流领域。 该公司使用开源LDAP面临的挑战 挑战1 开源…

【论文阅读】基于深度学习的时序异常检测——Anomaly Transformer

系列文章链接 论文一&#xff1a;2022 Anomaly Transformer 文章目录 理论概述代码实战 论文链接&#xff1a; Anomaly Transformer.pdf 代码链接&#xff1a; https://github.com/thuml/Anomaly-Transformer 视频讲解&#xff08;原作者禁止转载&#xff0c;联系的话侵删&am…

DC电源模块对于定制的要求主要有这几点

BOSHIDA DC电源模块对于定制的要求主要有这几点 DC电源模块是一种将交流电转换成为稳定的直流电的装置。在现代工业生产中&#xff0c;DC电源模块被广泛应用于各种电子设备中&#xff0c;例如计算机、手机、电视等。为了满足不同用户需求&#xff0c;DC电源模块的定制需求也是…

K8s中的Ingress

1.把端口号对外暴露&#xff0c;通过ip端口号进行访问 使用Service里面的NodePort实现 2.NodePort缺陷 在每个节点上都会起到端口&#xff0c;在访问时候通过任何节点&#xff0c;通过节点ip暴露端口号实现访问 意味着每个端口只能使用一次&#xff0c;一个端口对应一个应用…

pycharm中opencv库导入 cv2. 无函数提示跳出解决方法

pycharm中opencv库导入 cv2. 无函数提示跳出解决方法 1、找到当前解释器安装目录 例如&#xff1a; 2、进入D:\Python37\Lib\site-packages\cv2文件&#xff0c;进入cv2文件夹&#xff1a; 找到cv2.pyd, 把cv2.pyd复制一份&#xff0c;放到上层文件夹下&#xff0c;即site-p…

算法基础简介

目录 1、递归 2、二分查找 3、排序算法 分类 3.1、冒泡排序 3.2、选择排序 3.3、插入排序 3.4、希尔排序(高级插入排序) 3.5、归并排序 3.6、快速排序 核心思想 具体步骤 代码实现 3.7、堆排序 3.8、计数排序 3.9、桶排序 3.10、基数排序 4、字符串匹…

Pytest测试框架1

目录&#xff1a; 1.pytest简介、安装与准备2.pytest命名规则3.pycharm配置与界面化运行4.pytest测试用例结构5.pytest测试用例断言6.pytest测试框架结构7.计算器实战 1.pytest简介、安装与准备 前言 自动化测试前&#xff0c;需要提前准备好数据&#xff0c;测试完成后&am…

Linux下安装VMware虚拟机

目录 1. 简介 2. 工具/原料 2.1. 下载VMware 2.2. 安装 1. 简介 ​ VMware Workstation&#xff08;中文名“威睿工作站”&#xff09;是一款功能强大的桌面虚拟计算机软件&#xff0c;提供用户可在单一的桌面上同时运行不同的操作系统&#xff0c;和进行开发、测试 …

【Python数据容器】--- 列表的基本使用

个人主页&#xff1a;平行线也会相交 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 平行线也会相交 原创 收录于专栏【Python小白从入门到精通】&#x1f388; 本专栏旨在分享学习Python的一点学习心得&#xff0c;欢迎大家在评论区讨论&#x1f48c; 在学…

重磅发布|《人效九宫格:劳动力效能提升指引白皮书》,人效提升完全指南

5月&#xff0c;盖雅工场发布 「人效九宫格」 后引起了各位管理者和人力资源从业者的关注讨论。为方便大家更广泛地交流学习&#xff0c;在理论基础上&#xff0c;我们筹备了 《人效九宫格&#xff1a;劳动力效能提升指引白皮书》于今日正式发布&#xff0c;以更详细地阐述企…

Spring Boot如何整合mybatisplus

文章目录 1. 相关配置和代码2. 整合原理2.1 spring boot自动配置2.2 MybatisPlusAutoConfiguration2.3 debug流程2.3.1 MapperScannerRegistrar2.3.2MapperScannerConfigurer2.3.3 创建MybatisPlusAutoConfiguration2.3.4 创建sqlSessionFactory2.3.5 创建SqlSessionTemplate2.…

2023-08-07 LeetCode每日一题(反转字符串)

2023-08-07每日一题 一、题目编号 344. 反转字符串二、题目链接 点击跳转到题目位置 三、题目描述 编写一个函数&#xff0c;其作用是将输入的字符串反转过来。输入字符串以字符数组 s 的形式给出。 不要给另外的数组分配额外的空间&#xff0c;你必须原地修改输入数组、…

在centos7上使用非编译方式安装ffmpeg

很多在centos7上安装ffmpeg的教程都需要使用编译方式的安装&#xff1b;编译时间较长而且需要配置; 后来搜索到可以通过加载rpm 源的方式实现快速便捷操作 第一种方式&#xff1a; 首先需要安装yum源&#xff1a; yum install epel-release yum install -y https://mirrors.…