GitController.py 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601
  1. # -*- coding: <encoding name> -*-
  2. from git import Repo
  3. from os.path import split, exists
  4. import os
  5. import subprocess
  6. from time import time
  7. import random
  8. sys_seeting = dict(
  9. shell=True,
  10. stdin=subprocess.PIPE,
  11. stdout=subprocess.PIPE,
  12. stderr=subprocess.STDOUT,
  13. universal_newlines=True)
  14. git_path = 'git' # git的地址,如果配置了环境变量则不需要修改
  15. stop_key = '【操作完成】' # 存储stopKey的global变量
  16. passwd = '1234567890qwertyuiopasdfghjklzxcvbnmQWERTYUIOPASDFGHJKLZXCVBNM' # stopKey的候选词库
  17. class GitRepo: # git的基类
  18. def __init__(self, repo_dir, *args, **kwargs):
  19. self.url = None
  20. try:
  21. if exists(repo_dir + r'/.git'): # 是否为git 仓库
  22. pass
  23. elif repo_dir[-4:] == '.git':
  24. repo_dir = repo_dir[:-5] # -5,得把/去掉
  25. else:
  26. raise Exception
  27. except BaseException:
  28. subprocess.Popen(
  29. f'{git_path} init',
  30. cwd=self.repo_dir,
  31. **sys_seeting).wait()
  32. self.repo_dir = repo_dir # 仓库地址(末尾不带/)
  33. self.repo = Repo(repo_dir) # 读取一个仓库
  34. self.name = split(repo_dir)[-1]
  35. self.have_clone = True
  36. def make_stop_key(self): # 生成一个随机stopKey
  37. global stop_key, passwd
  38. code = ''
  39. for _ in range(8): # 八位随机数
  40. code += passwd[random.randint(0, len(passwd) - 1)] # 时间戳+8位随机数
  41. stop_key = (str(time()) + code).replace('.', '')
  42. return stop_key
  43. def get_flie_list(self, file_list, is_file=True, pat=' '):
  44. if file_list == '.':
  45. file = '..'
  46. else:
  47. file_ = []
  48. for i in file_list:
  49. if i[:len(self.repo_dir)] == self.repo_dir:
  50. file_.append(i[len(self.repo_dir) + 1:]) # +1是为了去除/
  51. if not is_file:
  52. return file_
  53. file = pat.join(file_)
  54. return file
  55. def dir_list(self, all=True):
  56. listfile = []
  57. if all:
  58. listfile += [
  59. f'[当前分支] {self.repo.active_branch} 工作区{"不" if self.repo.is_dirty() else ""}干净 -> {self.name}']
  60. listfile += [f'{"[配置文件]" if i == ".git" else "[未跟踪]"if i in self.repo.untracked_files else "[已跟踪]"} {i}'
  61. for i in os.listdir(self.repo_dir)]
  62. return listfile
  63. def add(self, file_list):
  64. file = self.get_flie_list(file_list)
  65. return subprocess.Popen(
  66. f'echo 添加文件... && {git_path} add {file} && echo {self.make_stop_key()}',
  67. cwd=self.repo_dir,
  68. **sys_seeting)
  69. def del_cached_file(self, file_list):
  70. file = self.get_flie_list(file_list)
  71. return subprocess.Popen(
  72. f'echo 撤销文件... && {git_path} rm --cached {file} && echo {self.make_stop_key()}',
  73. cwd=self.repo_dir,
  74. **sys_seeting)
  75. def commit_file(self, m):
  76. return subprocess.Popen(
  77. f'echo 提交文件: && {git_path} commit -m "{m}" && echo {self.make_stop_key()}',
  78. cwd=self.repo_dir,
  79. **sys_seeting)
  80. def status(self): # 执行status
  81. return subprocess.Popen(
  82. f'echo 仓库状态: && {git_path} status && echo {self.make_stop_key()}',
  83. cwd=self.repo_dir,
  84. **sys_seeting)
  85. def log(self, graph, pretty, abbrev): # 执行log
  86. args = ''
  87. if graph:
  88. args += ' --graph'
  89. if pretty:
  90. args += ' --pretty=oneline'
  91. if abbrev:
  92. args += ' --abbrev-commit'
  93. return subprocess.Popen(
  94. f'echo 仓库日志: && {git_path} log{args} && echo {self.make_stop_key()}',
  95. cwd=self.repo_dir,
  96. **sys_seeting)
  97. def do_log(self): # 执行reflog
  98. return subprocess.Popen(
  99. f'echo 操作记录: && {git_path} reflog && echo {self.make_stop_key()}',
  100. cwd=self.repo_dir,
  101. **sys_seeting)
  102. def diff(self, MASTER='HEAD'): # 执行diff
  103. return subprocess.Popen(
  104. f'echo 文件日志: && {git_path} diff {MASTER} && echo {self.make_stop_key()}',
  105. cwd=self.repo_dir,
  106. **sys_seeting)
  107. def reset(self, HEAD='HEAD~1', Type=0):
  108. if Type == 0:
  109. type_ = '--mixed' # 退回到工作区
  110. elif Type == 1:
  111. type_ = '--soft' # 退回到暂存区
  112. else:
  113. type_ = '--hard' # 退回到暂存区
  114. return subprocess.Popen(
  115. f'echo 回退... && {git_path} reset {type_} {HEAD} && echo {self.make_stop_key()}',
  116. cwd=self.repo_dir,
  117. **sys_seeting)
  118. def checkout(self, file_list):
  119. if len(file_list) >= 1: # 多于一个文件,不用--,空格
  120. file = self.get_flie_list(file_list, pat=' ')
  121. return subprocess.Popen(
  122. f'echo 丢弃修改: && {git_path} checkout {file} && echo {self.make_stop_key()}',
  123. cwd=self.repo_dir,
  124. **sys_seeting)
  125. elif len(file_list) == 1:
  126. return subprocess.Popen(
  127. f'echo 丢弃修改: && {git_path} checkout -- {file_list[0]} && echo {self.make_stop_key()}',
  128. cwd=self.repo_dir,
  129. **sys_seeting)
  130. else:
  131. return subprocess.Popen(
  132. f'echo 丢弃修改: && {git_path} checkout * && echo {self.make_stop_key()}',
  133. cwd=self.repo_dir,
  134. **sys_seeting)
  135. def rm(self, file_list): # 删除版本库中的文件
  136. file = self.get_flie_list(file_list)
  137. return subprocess.Popen(
  138. f'echo 删除... && {git_path} rm {file}',
  139. cwd=self.repo_dir,
  140. **sys_seeting)
  141. def branch_view(self): # 查看本地分支和远程分支
  142. return subprocess.Popen(
  143. f'echo 仓库分支: && {git_path} branch -a && echo 远程仓库信息: && {git_path} remote -v && '
  144. f'echo 分支详情: && {git_path} branch -vv && echo {self.make_stop_key()}',
  145. cwd=self.repo_dir,
  146. **sys_seeting)
  147. def new_branch(self, branch_name, origin): # 新建分支
  148. return subprocess.Popen(
  149. f'echo 新建分支... && {git_path} branch {branch_name} {origin} && echo {self.make_stop_key()}',
  150. cwd=self.repo_dir,
  151. **sys_seeting)
  152. def switch_branch(self, branch_name): # 切换分支
  153. return subprocess.Popen(
  154. f'echo 切换分支... && {git_path} switch {branch_name} && echo {self.make_stop_key()}',
  155. cwd=self.repo_dir,
  156. **sys_seeting)
  157. def del_branch(self, branch_name, del_type): # 删除分支
  158. return subprocess.Popen(
  159. f'echo 删除分支... && {git_path} branch -{del_type} {branch_name} && echo {self.make_stop_key()}',
  160. cwd=self.repo_dir,
  161. **sys_seeting)
  162. def merge_branch(self, branch_name, no_ff, m=''): # 合并分支
  163. if no_ff:
  164. no_ff = f' --no-ff -m "{m}"' # --no-ff前有空格
  165. else:
  166. no_ff = ''
  167. return subprocess.Popen(
  168. f'echo 合并分支... && {git_path} merge{no_ff} {branch_name} && echo {self.make_stop_key()}',
  169. cwd=self.repo_dir,
  170. **sys_seeting)
  171. def merge_abort(self): # 退出冲突处理
  172. return subprocess.Popen(
  173. f'echo 冲突处理退出... && {git_path} merge --abort && echo {self.make_stop_key()}',
  174. cwd=self.repo_dir,
  175. **sys_seeting)
  176. def save_stash(self): # 保存工作区
  177. return subprocess.Popen(
  178. f'echo 保存工作区... && {git_path} stash && echo {self.make_stop_key()}',
  179. cwd=self.repo_dir,
  180. **sys_seeting)
  181. def get_stash_list(self): # 工作区列表
  182. return subprocess.Popen(
  183. f'echo 工作区列表: && {git_path} stash list && echo {self.make_stop_key()}',
  184. cwd=self.repo_dir,
  185. **sys_seeting)
  186. def apply_stash(self, stash_num='0'): # 恢复工作区
  187. return subprocess.Popen(
  188. f'echo 恢复工作区... && {git_path} stash apply stash@{{{stash_num}}} && echo {self.make_stop_key()}',
  189. cwd=self.repo_dir,
  190. **sys_seeting)
  191. def drop_stash(self, stash_num='0'): # 删除工作区
  192. return subprocess.Popen(
  193. f'echo 删除工作区... && {git_path} stash drop stash@{{{stash_num}}} && echo {self.make_stop_key()}',
  194. cwd=self.repo_dir,
  195. **sys_seeting)
  196. def cherry_pick(self, commit): # 补丁
  197. return subprocess.Popen(
  198. f'echo 补丁... && {git_path} cherry-pick {commit} && echo {self.make_stop_key()}',
  199. cwd=self.repo_dir,
  200. **sys_seeting)
  201. def del_remote(self, remote_name):
  202. return subprocess.Popen(
  203. f'echo 删除远程仓库... && {git_path} remote remove {remote_name} && echo {self.make_stop_key()}',
  204. cwd=self.repo_dir,
  205. **sys_seeting)
  206. def remote_add(self, remote, remote_name):
  207. return subprocess.Popen(
  208. f'echo 添加远程仓库... && {git_path} remote add {remote_name} {remote} && echo {self.make_stop_key()}',
  209. cwd=self.repo_dir,
  210. **sys_seeting)
  211. def bind_branch(self, local_name, remote_name):
  212. return subprocess.Popen(
  213. f'echo 分支绑定... && {git_path} branch --set-upstream-to={remote_name} {local_name} && echo {self.make_stop_key()}',
  214. cwd=self.repo_dir,
  215. **sys_seeting)
  216. def push_tag(self, tag, remote_name):
  217. return subprocess.Popen(
  218. f'echo 推送标签... && {git_path} push {remote_name} {tag} && echo {self.make_stop_key()}',
  219. cwd=self.repo_dir,
  220. **sys_seeting)
  221. def del_tag(self, tag):
  222. return subprocess.Popen(
  223. f'echo 删除本地标签... && {git_path} tag -d {tag} && echo {self.make_stop_key()}',
  224. cwd=self.repo_dir,
  225. **sys_seeting)
  226. def add_tag(self, tag, commit, message=''):
  227. a = ' -a'
  228. if message != '':
  229. message = f' -m "{message}"' # 自带空格
  230. else:
  231. a = ''
  232. if commit != '':
  233. commit = f' {commit}' # 自带空格
  234. return subprocess.Popen(
  235. f'echo 添加标签... && {git_path} tag{a} {tag}{commit}{message} && echo {self.make_stop_key()}',
  236. cwd=self.repo_dir,
  237. **sys_seeting)
  238. def get_tag_list(self, condition=''):
  239. if condition != '':
  240. condition = f' -l {condition}'
  241. return subprocess.Popen(
  242. f'echo 标签列表: && {git_path} tag{condition} && echo {self.make_stop_key()}',
  243. cwd=self.repo_dir,
  244. **sys_seeting)
  245. def search_commit(self, condition):
  246. return subprocess.Popen(
  247. f'echo 查询结果: && {git_path} show {condition} && echo {self.make_stop_key()}',
  248. cwd=self.repo_dir,
  249. **sys_seeting)
  250. def pull_push(
  251. self,
  252. pull_or_push=0,
  253. remote='',
  254. remote_branch='',
  255. local='',
  256. allow=False,
  257. u=False,
  258. f=False):
  259. # 处理逻辑
  260. # 1)remote去斜杠第一个作为主机名字
  261. # 2) 从remote分离主机名(如果没指定)
  262. # 3) 如果local为空,用HEAD填充
  263. # 4) 如果以上后,主机名仍为空,则local和分支均为空
  264. split = remote.split('/')
  265. try:
  266. remote_name = split[0] # 获取主机名 1)
  267. except BaseException:
  268. remote_name = '' # 没有主机名 1)
  269. branch_split = remote_branch.split('/')
  270. if len(branch_split) >= 2 and remote_name == '':
  271. remote_name = branch_split[0] # 2)
  272. remote_branch = '/'.join(branch_split[1:]) # 2)
  273. if local.replace(' ', '') == '':
  274. local = 'HEAD' # 3)
  275. if remote_name == '': # 4)
  276. branch = ''
  277. else:
  278. if pull_or_push == 1:
  279. # 注意,local不可以为空,也不会为空
  280. if remote_branch != '':
  281. # git push <远程主机名> <本地分支名>:<远程分支名>
  282. branch = f'{local}:{remote_branch}'
  283. else:
  284. branch = f'{local}' # 要去掉冒号
  285. else:
  286. if remote_branch != 'HEAD':
  287. # git push <远程主机名> <本地分支名>:<远程分支名>
  288. branch = f'{remote_branch}:{local}'
  289. else:
  290. branch = f'{remote_branch}'
  291. if allow:
  292. history = ' --allow-unrelated-histories'
  293. else:
  294. history = ''
  295. push_pull = {
  296. 0: "pull",
  297. 1: f"push{' -u' if u else ''}{' -f' if f else ''}"}
  298. return subprocess.Popen(
  299. f'''echo 与服务器连接... && {git_path} {push_pull.get(pull_or_push, "pull")}{history} {remote_name} {branch}
  300. && echo {self.make_stop_key()}''',
  301. cwd=self.repo_dir,
  302. **sys_seeting)
  303. def del_branch_remote(self, remote, remote_branch):
  304. remote_split = remote.split('/')
  305. try:
  306. remote_name = remote_split[0] # 获取主机名 1)
  307. except BaseException:
  308. remote_name = '' # 没有主机名 1)
  309. branch_split = remote_branch.split('/')
  310. if len(branch_split) >= 2 and remote_name == '':
  311. remote_name = branch_split[0] # 2)
  312. remote_branch = '/'.join(branch_split[1:]) # 2)
  313. return subprocess.Popen(
  314. f'echo 删除远程分支... && {git_path} push {remote_name} :{remote_branch} && echo {self.make_stop_key()}',
  315. cwd=self.repo_dir,
  316. **sys_seeting)
  317. def del_tag_remote(self, remote, tag):
  318. return subprocess.Popen(
  319. f'echo 删除远程标签... && {git_path} push {remote} :refs/tags/{tag} && echo {self.make_stop_key()}',
  320. cwd=self.repo_dir,
  321. **sys_seeting)
  322. def fetch(self, remote, remote_branch, local):
  323. # 处理逻辑
  324. # 1)remote去斜杠第一个作为主机名字
  325. # 2) 从remote分离主机名(如果没指定)
  326. # 3) 如果local为空,用HEAD填充
  327. # 4) 如果以上后,主机名仍为空,则local和分支均为空
  328. split = remote.split('/')
  329. try:
  330. remote_name = split[0] # 获取主机名 1)
  331. except BaseException:
  332. remote_name = '' # 没有主机名 1)
  333. branch_split = remote_branch.split('/')
  334. if len(branch_split) >= 2 and remote_name == '':
  335. remote_name = branch_split[0] # 2)
  336. remote_branch = '/'.join(branch_split[1:]) # 2)
  337. if local.replace(' ', '') == '':
  338. local = 'HEAD' # 3)
  339. if remote_name == '': # 4)
  340. branch = ''
  341. else:
  342. if remote_branch != 'HEAD':
  343. # git push <远程主机名> <本地分支名>:<远程分支名>
  344. branch = f'{remote_branch}:{local}'
  345. else:
  346. branch = f'{remote_branch}'
  347. return subprocess.Popen(
  348. f'''echo 更新远程仓库... && {git_path} fetch {remote_name} {branch} && echo {self.make_stop_key()}''',
  349. cwd=self.repo_dir,
  350. **sys_seeting)
  351. def customize_command(self, command: str):
  352. return subprocess.Popen(f'{command} && echo {self.make_stop_key()}', cwd=self.repo_dir, **sys_seeting)
  353. def clone(self, url):
  354. return subprocess.Popen(
  355. f'echo 克隆操作不被允许 && echo {self.make_stop_key()}',
  356. cwd=self.repo_dir,
  357. **sys_seeting)
  358. def make_dir(self, dir):
  359. if len(dir) == '':
  360. return dir
  361. inside = '/'
  362. if dir[0] == '/':
  363. inside = ''
  364. return self.repo_dir + inside + dir
  365. def reset_file(self, hard, file_list): # 注意版本回退是:Reset_File
  366. file = self.get_flie_list(file_list)
  367. return subprocess.Popen(
  368. f'''echo 回退文件... && {git_path} reset {hard} {file} && echo {self.make_stop_key()}''',
  369. cwd=self.repo_dir, **sys_seeting)
  370. def rename_branch(self, old_name, new_name):
  371. return subprocess.Popen(
  372. f'''echo 回退文件... && {git_path} branch -m {old_name} {new_name} && echo {self.make_stop_key()}''',
  373. cwd=self.repo_dir, **sys_seeting)
  374. class CloneGit(GitRepo): # Clone一个git
  375. def __init__(self, Dic, *args, **kwargs):
  376. self.Repo_Dic = Dic # 仓库地址
  377. self.url = None
  378. self.name = split(Dic)[-1]
  379. self.have_clone = False
  380. def clone(self, url):
  381. if self.have_clone:
  382. super(CloneGit, self).clone(url)
  383. self.have_clone = True
  384. return subprocess.Popen(
  385. f'echo 正在克隆... && {git_path} clone {url} {self.Repo_Dic}',
  386. cwd=split(
  387. self.Repo_Dic)[0],
  388. **sys_seeting)
  389. def after_clone(self):
  390. self.repo = Repo(self.Repo_Dic)
  391. class git_Ctrol:
  392. def __init__(self):
  393. self.git_dict = {} # 名字-文件位置
  394. self.git_type_dict = {} # 名字-类型
  395. def open_repo(self, repo_dir, **kwargs):
  396. git = GitRepo(repo_dir)
  397. self.git_dict[git.name] = git
  398. self.git_type_dict[git.name] = 'init'
  399. return git.name
  400. def clone_repo(self, repo_dir, **kwargs):
  401. git = CloneGit(repo_dir)
  402. self.git_dict[git.name] = git
  403. self.git_type_dict[git.name] = 'clone'
  404. return git.name
  405. def get_git(self, name):
  406. return self.git_dict[name]
  407. def get_git_dict(self):
  408. return self.git_dict.copy()
  409. def get_dir(self, name):
  410. return self.get_git(name).dir_list()
  411. def add_file(self, name, file_list):
  412. return self.get_git(name).add(file_list)
  413. def del_cached_file(self, name, file_list):
  414. return self.get_git(name).del_cached_file(file_list) # 移除出去暂存区
  415. def commit_file(self, name, message):
  416. return self.get_git(name).commit_file(message)
  417. def log(self, name, graph, pretty, abbrev):
  418. return self.get_git(name).log(graph, pretty, abbrev)
  419. def do_log(self, name):
  420. return self.get_git(name).do_log()
  421. def status(self, name):
  422. return self.get_git(name).status()
  423. def diff_file(self, name, brach):
  424. return self.get_git(name).diff(brach)
  425. def back_version(self, name, head, reset_type=0):
  426. return self.get_git(name).reset(head, reset_type) # 版本回退HEAD
  427. def back_version_file(self, name, head, file_list):
  428. return self.get_git(name).reset_file(head, file_list) # 文件回退
  429. def checkout_version(self, name, file):
  430. return self.get_git(name).checkout(file) # 弹出
  431. def rm(self, name, file):
  432. return self.get_git(name).rm(file)
  433. def branch_view(self, name):
  434. return self.get_git(name).branch_view()
  435. def new_branch(self, name, new_branch, origin):
  436. return self.get_git(name).new_branch(new_branch, origin)
  437. def switch_branch(self, name, branch_name):
  438. return self.get_git(name).switch_branch(branch_name)
  439. def del_branch(self, name, branch_name, type_):
  440. d = {1: 'd', 0: 'D'}.get(type_, 'd')
  441. return self.get_git(name).del_branch(branch_name, d)
  442. def merge_branch(self, name, branch_name, no_ff, m):
  443. return self.get_git(name).merge_branch(branch_name, no_ff, m)
  444. def merge_abort(self, name):
  445. return self.get_git(name).merge_abort()
  446. def save_stash(self, name):
  447. return self.get_git(name).save_stash()
  448. def stash_list(self, name):
  449. return self.get_git(name).get_stash_list()
  450. def apply_stash(self, name, stash_num='0'):
  451. return self.get_git(name).apply_stash(stash_num)
  452. def drop_stash(self, name, stash_num='0'):
  453. return self.get_git(name).drop_stash(stash_num)
  454. def cherry_pick(self, name, commit):
  455. return self.get_git(name).cherry_pick(commit)
  456. def del_remote(self, name, remote_name):
  457. return self.get_git(name).del_remote(remote_name)
  458. def remote_add(self, name, remote, remote_name):
  459. return self.get_git(name).remote_add(remote, remote_name)
  460. def bind_branch(self, name, local_name, remote_name):
  461. return self.get_git(name).bind_branch(local_name, remote_name)
  462. def pull_from_remote(self, name, local_name, remote_name, remote_branch, allow=False, u=False):
  463. return self.get_git(name).pull_push(
  464. 0, remote_name, remote_branch, local_name, allow, u, False)
  465. def push_to_remote(self, name, local_name, remote_name, remote_branch, u=False, f=False):
  466. return self.get_git(name).pull_push(
  467. 1, remote_name, remote_branch, local_name, False, u, f) # push没有allow选项
  468. def tag(self, name, condition=''):
  469. return self.get_git(name).get_tag_list(condition) # push没有allow选项
  470. def show_new(self, name, condition):
  471. return self.get_git(name).search_commit(condition) # push没有allow选项
  472. def add_tag(self, name, tag, commit, message=''):
  473. return self.get_git(name).add_tag(
  474. tag, commit, message) # push没有allow选项
  475. def push_tag(self, name, tag, remoto):
  476. return self.get_git(name).push_tag(tag, remoto)
  477. def push_all_tag(self, name, remoto):
  478. return self.get_git(name).push_tag('--tags', remoto)
  479. def del_tag_remote(self, name, remote, tag):
  480. return self.get_git(name).del_tag_remote(remote, tag)
  481. def del_branch_remote(self, name, remote, remote_Branch):
  482. return self.get_git(name).del_branch_remote(remote, remote_Branch)
  483. def del_tag(self, name, tag):
  484. return self.get_git(name).del_tag(tag)
  485. def fetch(self, name, local_name, remote_name, remote_branch):
  486. return self.get_git(name).fetch(remote_name, remote_branch, local_name)
  487. def customize_command(self, name, command: str):
  488. return self.get_git(name).customize_command(command)
  489. def clone(self, name, url):
  490. return self.get_git(name).clone(url)
  491. def after_clone(self, name):
  492. try:
  493. return self.get_git(name).after_clone()
  494. except BaseException:
  495. return None
  496. def make_dir(self, name, dir):
  497. return self.get_git(name).make_dir(dir)
  498. def rename_branch(self, name, old_name, new_name):
  499. return self.get_git(name).rename_branch(old_name, new_name)