【Python自动化系列】批量提交文件到各个仓库并创建PR到指定分支
off999 2024-11-12 12:00 34 浏览 0 评论
一、问题
本地有一个文件夹openEuler-22.03-LTS(如下图所示),文件夹名是远程(Gitee平台)仓库的分支名,文件都是yaml文件,文件名对应仓库名,都是https://gitee.com/src-oepkgs/下的仓库,需要将每个文件依次添加到各自对应的仓库中,并提交PR到远程仓库的openEuler-22.03-LTS分支上。
二、思路1:走fork进行合并
先fork后clone,PR在相同的分支不同的用户(fork和被fork的用户)之间
Step1
由文件名解析出仓库名,与https://gitee.com/src-oepkgs/进行拼接,得到仓库的完整地址,比如:https://gitee.com/src-oepkgs/a2jmidid.git,为例
Step2
这个仓库fork到自己的账号(自己也当成组织),此时这个仓库的归属权是我,这个仓库的地址变为https://gitee.com/oepkgs-support/a2jmidid
Step3
将fork后(归属权是我)的仓库https://gitee.com/oepkgs-support/a2jmidid.git克隆指定分支(openEuler-22.03-LTS分支)到本地(电脑)D:/workspace/自动化/openEuler-22.03-LTS文件夹下,D:/workspace/自动化/openEuler-22.03-LTS/a2jmidid
git clone -b openEuler-22.03-LTS https://gitee.com/oepkgs-support/a2jmidid.gitStep4
进入到这个仓库文件夹,会自动会切换到本地的openEuler-22.03-LTS分支
Step5
取回远程主机openEuler-22.03-LTS分支的更新,再与本地的openEuler-22.03-LTS分支合并
git pull origin openEuler-22.03-LTS: openEuler-22.03-LTSStep6
将openEuler-22.03-LTS文件夹下的a2jmidid.yaml文件复制到这个项目仓库(文件夹:D:/workspace/自动化/openEuler-22.03-LTS / a2jmidid)中
Step7
提交并推送到远程的openEuler-22.03-LTS分支
git add -A
git commit -m ‘add files’
git push origin openEuler-22.03-LTS:openEuler-22.03-LTS这里直接push到远程的openEuler-22.03-LTS分支是可以的,因为这个push是在本地oepkgs-support/a2jmidid和远程的oepkgs-support/a2jmidid这同一个用户之间。oepkgs-support(人)是这个仓库的所有者,有直接推送这个权限。
Step8
创建oepkgs-support:openEuler-22.03-LTS这个分支到src-oepkgs:openEuler-22.03-LTS分支的PR:https://gitee.com/src-oepkgs/a2jmidid/pulls/1
完整代码:
import os
import subprocess
import time
import requests
import shutil
import pandas as pd
def fork_repo(repo_name):
# # 仓库信息
owner = "src-oepkgs" # 替换为要Fork的用户名
# 授权信息
access_token = 'xxxxxxxxxxx' # 替换为自己账号的访问令牌
# API URL
url = f"https://gitee.com/api/v5/repos/{owner}/{repo_name}/forks"
# 请求头
headers = {
"Content-Type": "application/json",
"Authorization": f"Bearer {access_token}"
}
# 发送POST请求进行Fork操作
response = requests.post(url, headers=headers)
# 检查响应状态码
if response.status_code == 201:
print("Fork created successfully!")
print("Forked repository URL:", response.json()["html_url"])
else:
print("Failed to create Fork.")
print("Error message:", response.json()["message"])
def delete_repository(repo_name):
owner = "oepkgs-support" # 替换仓库所属空间地址
access_token = 'xxxxxxxxxxx' # 替换为你的访问令牌
url = f"https://gitee.com/api/v5/repos/{owner}/{repo_name}"
headers = {
"Authorization": f"Bearer {access_token}"
}
response = requests.delete(url, headers=headers)
if response.status_code == 204:
print("仓库删除成功")
else:
print("仓库删除失败")
def git_add_commit_push(repo_path, file_name, branch):
"""执行 git add, commit, push"""
os.chdir(repo_path)
# subprocess.run(["git", "pull", "origin", branch, ":", branch])
subprocess.run(["git", "add", file_name])
subprocess.run(["git", "commit", "-m", f"Add {file_name}"])
subprocess.run(["git", "push", "origin", f"{branch}:{branch}"])
def create_pr(repo_name):
# 仓库信息
owner = "src-oepkgs" # 替换仓库所属空间地址
# 授权信息
access_token = 'xxxxxxxxxxxxx' # 替换为自己账号的访问令牌
# 创建Pull Request的参数
title = "add yaml file"
head = "oepkgs-support:openEuler-22.03-LTS" # 替换为您的源分支名称
base = "openEuler-22.03-LTS" # 替换为您的目标分支名称
# API URL
url = f"https://gitee.com/api/v5/repos/{owner}/{repo_name}/pulls"
# 请求头
headers = {
"Content-Type": "application/json",
"Authorization": f"Bearer {access_token}"
}
# 请求体
data = {
"title": title,
"head": head,
"base": base
}
# 发送POST请求创建Pull Request
response = requests.post(url, headers=headers, json=data)
# 检查响应状态码
if response.status_code == 201:
print("Pull Request created successfully!")
print("Pull Request URL:", response.json()["html_url"])
else:
print("Failed to create Pull Request.")
print("Error message:", response.json()["message"])
def main():
# 本地文件夹路径和远程分支名
basedir = os.path.abspath(os.path.dirname(os.path.dirname(__file__)))
local_folder = os.path.join(basedir, "openEuler-22.03-LTS") # 本地文件夹路径
branch_name = "openEuler-22.03-LTS"
index = 0
abnormal_repo = []
start_ = 'a'
end_ = 'z'
# 遍历文件夹中的每个文件
for file_name in os.listdir(local_folder):
# if file_name.endswith('.yaml'):
if file_name.lower().endswith('.yaml') and start_ <= file_name[0].lower() <= end_:
# 解析出文件名
repo_name = file_name.replace('.yaml', '') # 从文件名获取仓库名
print(repo_name)
index = index + 1
# fork仓库
try:
fork_repo(repo_name)
fork_repo_url = f"https://gitee.com/oepkgs-support/{repo_name}.git"
# 克隆仓库(如果需要)
repo_path = os.path.join(local_folder, repo_name)
if not os.path.exists(repo_path):
try:
subprocess.run(["git", "clone", "-b", branch_name, fork_repo_url, repo_path])
except Exception:
abnormal_repo.append(repo_name)
else:
# 判断repopath这个文件夹存不存在
if os.path.exists(repo_path):
print('clone success')
# 先pull一下,取回远程主机openEuler-22.03-LTS分支的更新
os.chdir(repo_path)
subprocess.run(["git", "config", "--global", "http.sslVerify", "true"])
subprocess.run(["git", "pull", "origin", branch_name])
# 将文件复制到仓库目录中
file_path = os.path.join(local_folder, file_name)
shutil.copy(file_path, repo_path)
# 执行 git 操作
git_add_commit_push(repo_path, file_name, branch_name)
# 创建PR
create_pr(repo_name)
print(repo_path)
try:
# os.system(f"attrib -r {repo_path}")
# shutil.rmtree(repo_path)
# remove_readonly_dir(repo_path)
# os.removedirs(repo_path)
subprocess.run(["rmdir", "/s", "/q", repo_path], shell=True)
except Exception as e:
print("删除文件夹时出现错误:", str(e))
else:
abnormal_repo.append(repo_name)
except Exception as e:
print("fork出现错误:", str(e))
abnormal_repo.append(repo_name)
time.sleep(30)
fork_repo(repo_name)
if index % 100 == 0:
print(abnormal_repo)
df = pd.DataFrame(abnormal_repo, columns=['异常仓库'])
df.to_excel(os.path.join(basedir, 'log') + '\\' + f'{start_}-{end_}异常仓库{index}.xlsx', index=False)
if __name__ == "__main__":
main()问题1:fork不成功
fork出现错误: HTTPSConnectionPool(host='gitee.com', port=443): Max retries exceeded with url: /api/v5/repos/src-oepkgs/gtk-theme-adwaita-tweaks/forks (Caused by SSLError(SSLEOFError(8, 'EOF occurred in violation of protocol (_ssl.c:1129)')))
解决方法:引入异常捕获,增加时延,记录异常的仓库
问题2:fork成功但是不一定能够克隆下来
git clone时fatal: unable to access ‘https://gitee.com/XXX.git/‘: The requested returned error: 500
解决方法:引入异常捕获,记录异常的仓库
问题3:每个账号fork的项目数量有限制,最多1k,如果创建的PR没有合并,删除这个fork的仓库,这个PR也会被删除
换一种思路,走分支合并,不走fork合并。
三、走分支合并
不fork,直接clone原仓库,在远程源仓库创建新分支,走分支合并,PR合并在同一个用户的不同分支之间。
Step1
直接克隆原仓库到本地:git clone -b openEuler-22.03-LTS https://gitee.com/src-oepkgs/python-zeroconf.git
step2
进入到这个仓库文件夹,自动会切换到本地的openEuler-22.03-LTS分支
step3
取回远程主机openEuler-22.03-LTS分支的更新,再与本地的openEuler-22.03-LTS分支合并
git pull origin openEuler-22.03-LTS: openEuler-22.03-LTSstep4
将openEuler-22.03-LTS文件夹下的a2jmidid.yaml文件复制到这个项目仓库中
step5
提交并推送到远程的openEuler-22.03-LTS-yaml分支(相当于在远程建了一个新的openEuler-22.03-LTS-yaml分支并push到这个分支)
git add -A
git commit -m ‘add files’
git push origin openEuler-22.03-LTS:openEuler-22.03-LTS-yaml这里直接push到远程的openEuler-22.03-LTS是会报错的,因为这个push是在本地oepkgs/a2jmidid和远程的oepkgs/a2jmidid这同一个用户之间,oepkgs-support(人)不是仓库a2jmidid的管理员(所有者或开发者),没有直接推送这个权限,万一提交不当会直接污染这个远程分支,因此需要在远程新建一个分支push到这个新分支,然后创建openEuler-22.03-LTS-yaml这个分支到openEuler-22.03-LTS分支的PR进行合并。
step6
创建src-oepkgs:openEuler-22.03-LTS-yaml这个分支到src-oepkgs:openEuler-22.03-LTS分支的PR
完整代码:
import os
import subprocess
import time
import requests
import shutil
import pandas as pd
def git_add_commit_push(repo_path, file_name, branch):
"""执行 git add, commit, push"""
os.chdir(repo_path)
subprocess.run(['git', 'config', '--global', 'core.autocrlf', 'true'])
subprocess.run(["git", "add", file_name])
subprocess.run(["git", "commit", "-m", f"Add {file_name}"])
subprocess.run(["git", "push", "origin", f"{branch}:{branch}-yaml"])
def create_pr(repo_name):
# 仓库信息
owner = "src-oepkgs" # 替换仓库所属空间地址
# 授权信息
access_token = '4e8xxxx250afc5a19' # 自己账号的访问令牌
# 创建Pull Request的参数
title = "add yaml file"
head = "openEuler-22.03-LTS-yaml" # 源分支名称
base = "openEuler-22.03-LTS" # 目标分支名称
# API URL
url = f"https://gitee.com/api/v5/repos/{owner}/{repo_name}/pulls"
# 请求头
headers = {
"Content-Type": "application/json",
"Authorization": f"Bearer {access_token}"
}
# 请求体
data = {
"title": title,
"head": head,
"base": base
}
# 发送POST请求创建Pull Request
response = requests.post(url, headers=headers, json=data)
# 检查响应状态码
if response.status_code == 201:
print("Pull Request created successfully!")
print("Pull Request URL:", response.json()["html_url"])
else:
print("Failed to create Pull Request.")
print("Error message:", response.json()["message"])
def main():
# 本地文件夹路径和远程分支名
basedir = os.path.abspath(os.path.dirname(os.path.dirname(__file__)))
local_folder = os.path.join(basedir, "openEuler-22.03-LTS") # 本地文件夹路径
branch_name = "openEuler-22.03-LTS"
index = 0
abnormal_repo = []
start_ = 'a'
end_ = 'z'
# 遍历文件夹中的每个文件
for file_name in os.listdir(local_folder):
if file_name.lower().endswith('.yaml') and start_ <= file_name[0].lower() <= end_:
# 解析出文件名
repo_name = file_name.replace('.yaml', '') # 从文件名获取仓库名
print(repo_name)
index = index + 1
# 克隆仓库
repo_url = f"https://gitee.com/src-oepkgs/{repo_name}.git"
repo_path = os.path.join(local_folder, repo_name)
if not os.path.exists(repo_path):
try:
subprocess.run(["git", "clone", "-b", branch_name, repo_url, repo_path])
except Exception:
abnormal_repo.append(repo_name)
else:
# 判断repopath这个文件夹存不存在
if os.path.exists(repo_path):
print('clone success')
# 先pull一下,取回远程主机openEuler-22.03-LTS分支的更新
os.chdir(repo_path)
subprocess.run(["git", "config", "--global", "http.sslVerify", "true"])
subprocess.run(["git", "pull", "origin", branch_name])
# 将文件复制到仓库目录中
file_path = os.path.join(local_folder, file_name)
shutil.copy(file_path, repo_path)
# 执行 git 操作
git_add_commit_push(repo_path, file_name, branch_name)
# 创建PR
try:
create_pr(repo_name)
except Exception as e:
print("create pr出现错误:", str(e))
abnormal_repo.append(repo_name)
time.sleep(30)
create_pr(repo_name)
# 删除克隆的仓库文件夹内容
try:
subprocess.run(["rmdir", "/s", "/q", repo_path], shell=True)
except Exception as e:
print("删除文件夹时出现错误:", str(e))
else:
abnormal_repo.append(repo_name)
if index % 100 == 0:
print(abnormal_repo)
df = pd.DataFrame(abnormal_repo, columns=['异常仓库'])
df.to_excel(os.path.join(basedir, 'log') + '\\' + f'{start_}-{end_}异常仓库{index}.xlsx', index=False)
if __name__ == "__main__":
main()本人读研期间发表6篇SCI数据挖掘相关论文,现在某研究院从事数据算法相关科研工作,对Python有一定认知和理解,会结合自身科研实践经历不定期分享关于Python、机器学习、深度学习、人工智能基础知识与应用案例。
致力于只做原创,以最简单的方式理解和学习,关注我一起交流成长。
1、关注“数据杂坛”公众号,即可在后台联系我获取相关数据集和源码。
2、关注“数据杂坛”公众号,点击“领资料”即可免费领取资料书籍。
3、需要论文指导或商务合作,点击“联系我”添加作者微信直接交流。
原文链接:
相关推荐
- 已经停服的手游(已停服游戏大全)
-
停服了战区现在也已经开始进入到一个短暂的停服阶段,具体的原因主要是根据官方所说,这本身就是为了能够有效保持一种服务器的稳定,然后就会选择停服的操作。烈火如歌手游停运的原因是因为官方计划对游戏进行重制。...
- cad官方正版下载(cad官方正版下载入口)
-
要下载CAD电脑软件,首先需要找到可靠的CAD软件提供商的官方网站。在官方网站上,您可以找到下载CAD软件的链接。点击链接后,您将被引导到下载页面,选择适合您操作系统的版本。然后,点击下载按钮开始下载...
- pptv下载(pptv下载速度慢怎么解决)
-
步骤/方法1、请到http://www.pptv.com首页,点击“免费下载”2、点击“保存“3、选择存放“PPLive网络电视”的位置,点击“保存”4、等待“PPTV网络电视”下载完成5、下载完成...
- 群发助手软件(群发助手软件怎么用)
-
很多微友一般的做法都是建立微群。而微信群的人数也有限制,同时是多人的聊天,谈不上群发消息,只能说是群聊。如果使用微信自带的群发助手就可以实现一对多的群发消息给好友,那么微信群发助手在哪里呢?下面就...
- google store(googlestore应用商店)
-
很好,我无法为您提供谷歌商店网页版的登录入口。因为谷歌商店是谷歌公司旗下的在线购物平台,仅支持访问外网进行访问,在国内无法直接访问。同时,谷歌商店也并未提供官方登录入口。如果您需要访问谷歌商店,建议您...
-
- 在线ps网页版(在线ps网页版手机免费使用)
-
1.PS在线是指在互联网上使用PS(Photoshop)软件进行图像处理和编辑的一种方式。2.原因是PS是一款功能强大的图像处理软件,可以进行各种图像编辑、修饰、合成等操作,因此在互联网上提供了在线版本,方便用户随时随地进行图像处理。3...
-
2026-01-17 03:51 off999
- xy苹果助手下载手机版(xy苹果助手ios版)
-
1、先在没越狱的iPhone手机上下载XY苹果助手。2、点击打开XY苹果助手并信任软件。3、在XY苹果助手的必备里点击下载按钮下载安装必备软件应用。4、特定的具体软件应用,可以在XY苹果助手里使用搜索...
- 魔兽争霸3冰封王座官网下载(魔兽争霸3冰封王座1.20e中文版下载_魔兽争霸3下载)
-
没法下载,因为没有说明在什么平台下载可以在网易的官方网站上进行下载,下载地址如下:https://dz.blizzard.cn/下载的时候请选择中间的“平台完整版”,这个版本包含了《魔兽争霸3》游戏本...
- 四人麻将(四人麻将图片)
-
1.标准打法:常规的打缺打法,不附加任何的特殊规则。2.标准下雨:在标准打法的基础上,加入下雨的功能,也就是杠牌。下雨分为明杠和暗杠。明杠:①玩家手中有三张一样的牌,其它玩家打出了第四张一样的牌,...
- 美图秀秀app下载安装免费(美美秀图手机版下载)
-
美图秀秀下载了找不到,应该是因为你的内存的原因,安装失败了。美图秀秀是一款使用很普遍的美图软件,可以满足大多数用户的需求,比如自己p图,或者是拼图都很好。可是,你发现你下载了美图秀秀,却在手机桌面没有...
- 街机游戏厅app(街机游戏厅app大全)
-
以下是几个不错的iOS街机游戏厅:MAME4iOS:这是一款非常流行的iOS街机游戏厅,可以让你在iPhone或iPad上运行多个街机游戏。iCade:这是一款专门为iCade控制器设计的街机游戏...
- 海外加速器下载(加速器中国免费)
-
迅游不能下载海外游戏的原因是因为一些海外游戏需要进行版权验证,而迅游没有获得这些游戏的版权许可。此外,由于不同国家的游戏法规和政策不同,迅游有时需要根据不同的规定进行调整,这也可能会限制用户下载一些海...
- 日语输入法app下载(simeji日语输入法官网下载)
-
谷歌输入法。安卓可以用的最好用的了吧,在设置里下载好所需要语言,打字时按那个切换即可。另外还有一个触宝,个人觉得不如谷歌。可以打出日文的输入法,很不错的日语输入法。安装后请在系统中添加输入法,即可使...
-
- 字体设计网站(字体设计 网站)
-
那就到这个网站,每次只能输入一个字,从小篆到甲骨文:http://www.internationalscientific.org就玩个静态页,很简单啊,教你一个作弊方法(反正你来这里讨教也不是自己完成作业),就是找一个你认为合适你作业要求...
-
2026-01-17 01:51 off999
欢迎 你 发表评论:
- 一周热门
-
-
抖音上好看的小姐姐,Python给你都下载了
-
全网最简单易懂!495页Python漫画教程,高清PDF版免费下载
-
飞牛NAS部署TVGate Docker项目,实现内网一键转发、代理、jx
-
Python 3.14 的 UUIDv6/v7/v8 上新,别再用 uuid4 () 啦!
-
python入门到脱坑 输入与输出—str()函数
-
Python三目运算基础与进阶_python三目运算符判断三个变量
-
(新版)Python 分布式爬虫与 JS 逆向进阶实战吾爱分享
-
失业程序员复习python笔记——条件与循环
-
系统u盘安装(win11系统u盘安装)
-
Python 批量卸载关联包 pip-autoremove
-
- 最近发表
- 标签列表
-
- python计时 (73)
- python安装路径 (56)
- python类型转换 (93)
- python进度条 (67)
- python吧 (67)
- python的for循环 (65)
- python格式化字符串 (61)
- python静态方法 (57)
- python列表切片 (59)
- python面向对象编程 (60)
- python 代码加密 (65)
- python串口编程 (77)
- python封装 (57)
- python写入txt (66)
- python读取文件夹下所有文件 (59)
- python操作mysql数据库 (66)
- python获取列表的长度 (64)
- python接口 (63)
- python调用函数 (57)
- python多态 (60)
- python匿名函数 (59)
- python打印九九乘法表 (65)
- python赋值 (62)
- python异常 (69)
- python元祖 (57)
