1
0

refactor(scripts): 开发分支初始化脚本从 bash 重构为 Python,增强跨平台支持和用户体验

This commit is contained in:
2026-04-23 10:43:59 +08:00
parent b3258e76df
commit 58ebcaa299
2 changed files with 354 additions and 18 deletions

354
scripts/init/init-dev-branch.py Executable file
View File

@@ -0,0 +1,354 @@
#!/usr/bin/env python3
"""开发分支工作区初始化脚本
用于创建基于远端分支或新建的开发分支工作区。
"""
import argparse
import subprocess
import sys
from pathlib import Path
def run_git(args: list[str]) -> subprocess.CompletedProcess:
"""执行 git 命令
Args:
args: git 命令参数列表
Returns:
subprocess.CompletedProcess 对象
"""
return subprocess.run(
["git"] + args,
capture_output=True,
text=True,
check=False
)
def get_root_dir() -> Path:
"""获取 git 仓库根目录
Raises:
RuntimeError: 不在 git 仓库中
"""
result = run_git(["rev-parse", "--show-toplevel"])
if result.returncode != 0:
raise RuntimeError("不在 git 仓库中")
return Path(result.stdout.strip())
def fetch_remote() -> bool:
"""从远端获取最新分支信息
Returns:
是否成功获取
"""
result = run_git(["fetch", "--quiet"])
if result.returncode != 0:
print("警告: 无法获取远端信息,继续使用本地数据")
return False
return True
def get_all_remote_branches() -> list[str]:
"""获取所有远端分支列表
Returns:
远端分支列表
"""
result = run_git(["branch", "-r"])
if result.returncode != 0:
return []
return [line.strip() for line in result.stdout.strip().split("\n") if line.strip()]
def get_remote_branches(branch_name: str) -> list[str]:
"""获取远端匹配的分支列表
Args:
branch_name: 分支名称
Returns:
匹配的远端分支列表
"""
result = run_git(["branch", "-r"])
if result.returncode != 0:
return []
return [
line.strip()
for line in result.stdout.strip().split("\n")
if line.strip() and line.strip().endswith(f"/{branch_name}")
]
def branch_exists_local(branch_name: str) -> bool:
"""检查本地分支是否存在
Args:
branch_name: 分支名称
Returns:
是否存在
"""
result = run_git(["show-ref", "--verify", "--quiet", f"refs/heads/{branch_name}"])
return result.returncode == 0
def worktree_exists(worktree_path: Path) -> bool:
"""检查工作区是否存在
Args:
worktree_path: 工作区路径
Returns:
是否存在
"""
result = run_git(["worktree", "list"])
if result.returncode != 0:
return False
return str(worktree_path) in result.stdout
def extract_branch_name(remote_branch: str) -> str:
"""从远端分支名提取分支名
Args:
remote_branch: 远端分支名(如 origin/feature-123
Returns:
分支名(如 feature-123
"""
return remote_branch.split("/", 1)[1] if "/" in remote_branch else remote_branch
def validate_branch_creation(branch_name: str, worktree_path: Path) -> None:
"""验证是否可以创建分支和工作区
Args:
branch_name: 分支名称
worktree_path: 工作区路径
Raises:
RuntimeError: 验证失败
"""
if worktree_path.exists():
raise RuntimeError(f"工作区已存在于 {worktree_path}")
if worktree_exists(worktree_path):
raise RuntimeError(f"工作区 '{branch_name}' 已存在")
if branch_exists_local(branch_name):
raise RuntimeError(f"本地分支 '{branch_name}' 已存在")
def select_from_list(items: list[str], prompt: str, allow_create: bool = False) -> str | None:
"""让用户从列表中选择
Args:
items: 选项列表
prompt: 提示信息
allow_create: 是否允许创建新分支
Returns:
用户选择的项,或 None 表示创建新分支
"""
if not items:
return None
print(prompt)
for i, item in enumerate(items, 1):
print(f" {i}\t{item}")
if allow_create:
print(f" {len(items) + 1}\t创建新分支")
print()
max_choice = len(items) + 1 if allow_create else len(items)
while True:
try:
selection = int(input(f"请选择 (1-{max_choice}): "))
if 1 <= selection <= len(items):
selected = items[selection - 1]
print(f"已选择: {selected}")
return selected
if allow_create and selection == len(items) + 1:
return None
print(f"错误: 请输入 1-{max_choice} 之间的数字")
except ValueError:
print("错误: 请输入有效的数字")
except KeyboardInterrupt:
print("\n操作已取消")
sys.exit(1)
def input_branch_name() -> str:
"""让用户输入分支名称
Returns:
分支名称
"""
while True:
try:
name = input("请输入新分支名称: ").strip()
if name:
return name
print("错误: 分支名称不能为空")
except KeyboardInterrupt:
print("\n操作已取消")
sys.exit(1)
def create_worktree(branch_name: str, worktree_path: Path, base_branch: str | None = None) -> None:
"""创建工作区
Args:
branch_name: 新分支名称
worktree_path: 工作区路径
base_branch: 基础分支(可选)
Raises:
RuntimeError: 创建失败
"""
args = ["worktree", "add", "-b", branch_name, str(worktree_path)]
if base_branch:
args.append(base_branch)
result = run_git(args)
if result.returncode != 0:
raise RuntimeError(f"创建工作区失败: {result.stderr.strip()}")
def create_worktree_with_validation(
branch_name: str,
worktree_path: Path,
base_branch: str | None = None
) -> None:
"""验证并创建工作区
Args:
branch_name: 新分支名称
worktree_path: 工作区路径
base_branch: 基础分支(可选)
"""
validate_branch_creation(branch_name, worktree_path)
create_worktree(branch_name, worktree_path, base_branch)
print(f"工作区已创建于 {worktree_path}")
def handle_with_branch_name(branch_name: str, worktrees_dir: Path) -> None:
"""处理有分支名参数的情况
Args:
branch_name: 分支名称
worktrees_dir: 工作区目录
"""
worktree_path = worktrees_dir / branch_name
try:
validate_branch_creation(branch_name, worktree_path)
except RuntimeError as e:
print(f"错误: {e}")
sys.exit(1)
remote_branches = get_remote_branches(branch_name)
try:
if remote_branches:
selected_branch = select_from_list(
remote_branches,
"找到远端分支:",
allow_create=True
)
if selected_branch:
create_worktree(branch_name, worktree_path, selected_branch)
else:
create_worktree(branch_name, worktree_path)
else:
print("未找到远端分支,创建新分支")
create_worktree(branch_name, worktree_path)
print(f"工作区已创建于 {worktree_path}")
except RuntimeError as e:
print(f"错误: {e}")
sys.exit(1)
def handle_without_branch_name(worktrees_dir: Path) -> None:
"""处理无分支名参数的情况
Args:
worktrees_dir: 工作区目录
"""
remote_branches = get_all_remote_branches()
if not remote_branches:
print("未找到远端分支")
sys.exit(1)
selected_branch = select_from_list(
remote_branches,
"远端分支列表:",
allow_create=True
)
if selected_branch:
branch_name = extract_branch_name(selected_branch)
worktree_path = worktrees_dir / branch_name
try:
create_worktree_with_validation(branch_name, worktree_path, selected_branch)
except RuntimeError as e:
print(f"错误: {e}")
sys.exit(1)
else:
branch_name = input_branch_name()
worktree_path = worktrees_dir / branch_name
try:
create_worktree_with_validation(branch_name, worktree_path)
except RuntimeError as e:
print(f"错误: {e}")
sys.exit(1)
def main() -> None:
parser = argparse.ArgumentParser(
description="创建开发分支工作区",
formatter_class=argparse.RawDescriptionHelpFormatter,
epilog="""
示例:
%(prog)s # 列出远端分支供选择
%(prog)s feature-123 # 创建 feature-123 工作区
%(prog)s bugfix-456 # 创建 bugfix-456 工作区
"""
)
parser.add_argument("branch_name", nargs="?", help="分支名称(可选)")
args = parser.parse_args()
try:
root_dir = get_root_dir()
except RuntimeError as e:
print(f"错误: {e}")
sys.exit(1)
worktrees_dir = root_dir / ".worktrees"
worktrees_dir.mkdir(parents=True, exist_ok=True)
print("正在从远端获取最新分支信息...")
fetch_remote()
if args.branch_name:
handle_with_branch_name(args.branch_name, worktrees_dir)
else:
handle_without_branch_name(worktrees_dir)
if __name__ == "__main__":
main()

View File

@@ -1,18 +0,0 @@
#!/bin/bash
set -euo pipefail
if [ $# -eq 0 ]; then
echo "Usage: $0 <branch-name>"
exit 1
fi
BRANCH_NAME="$1"
ROOT_DIR=$(git rev-parse --show-toplevel)
WORKTREES_DIR="$ROOT_DIR/.worktrees"
mkdir -p "$WORKTREES_DIR"
git worktree add -b "$BRANCH_NAME" "$WORKTREES_DIR/$BRANCH_NAME"
echo "Worktree created at $WORKTREES_DIR/$BRANCH_NAME"