-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathcanary.py
122 lines (96 loc) · 4.14 KB
/
canary.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
import argparse
import re
import subprocess
from typing import Optional
# ファイルを読み込み、バージョンを更新
def update_version(file_path: str, dry_run: bool) -> Optional[str]:
with open(file_path, "r", encoding="utf-8") as f:
content: str = f.read()
# 現在のバージョンを取得
current_version_match = re.search(r'version\s*=\s*"([\d\.\w]+)"', content)
if not current_version_match:
raise ValueError("Version not found or incorrect format in pyproject.toml")
current_version: str = current_version_match.group(1)
# バージョンが .devX を持っている場合の更新
if ".dev" in current_version:
new_content, count = re.subn(
r'(version\s*=\s*")(\d+\.\d+\.\d+\.dev)(\d+)',
lambda m: f"{m.group(1)}{m.group(2)}{int(m.group(3)) + 1}",
content,
)
else:
# .devX がない場合、次のマイナーバージョンにして .dev0 を追加
new_content, count = re.subn(
r'(version\s*=\s*")(\d+)\.(\d+)\.(\d+)',
lambda m: f"{m.group(1)}{m.group(2)}.{int(m.group(3)) + 1}.0.dev0",
content,
)
if count == 0:
raise ValueError("Version not found or incorrect format in pyproject.toml")
# 新しいバージョンを確認
new_version_match = re.search(r'version\s*=\s*"([\d\.\w]+)"', new_content)
if not new_version_match:
raise ValueError("Failed to extract the new version after the update.")
new_version: str = new_version_match.group(1)
print(f"Current version: {current_version}")
print(f"New version: {new_version}")
confirmation: str = input("Do you want to update the version? (Y/n): ").strip().lower()
if confirmation != "y":
print("Version update canceled.")
return None
# Dry-run 時の動作
if dry_run:
print("Dry-run: Version would be updated to:")
print(new_content)
else:
with open(file_path, "w", encoding="utf-8") as f:
f.write(new_content)
print(f"Version updated in pyproject.toml to {new_version}")
return new_version
# uv sync を実行し、uv.lock を git に追加
def run_uv_sync(dry_run: bool) -> None:
if dry_run:
print("Dry-run: Would run 'uv sync' and add 'uv.lock' to git")
else:
# uv sync の実行
result = subprocess.run(["uv", "sync"], check=True, capture_output=True, text=True)
print(result.stdout)
# uv.lock ファイルを git に追加
subprocess.run(["git", "add", "uv.lock"], check=True)
print("Added 'uv.lock' to git")
# git コミット、タグ、プッシュを実行
def git_operations(new_version: str, dry_run: bool) -> None:
if dry_run:
print("Dry-run: Would run 'git add pyproject.toml' and 'git add uv.lock'")
print(f"Dry-run: Would run 'git commit -m Bump version to {new_version}'")
print(f"Dry-run: Would run 'git tag {new_version}'")
print("Dry-run: Would run 'git push'")
print("Dry-run: Would run 'git push --tags'")
else:
subprocess.run(["git", "add", "pyproject.toml"], check=True)
subprocess.run(
["git", "commit", "-m", f"[canary] Bump version to {new_version}"], check=True
)
subprocess.run(["git", "tag", new_version], check=True)
subprocess.run(["git", "push"], check=True)
subprocess.run(["git", "push", "--tags"], check=True)
# メイン処理
def main() -> None:
parser = argparse.ArgumentParser(
description="Update pyproject version, run uv sync, and commit changes."
)
parser.add_argument(
"--dry-run", action="store_true", help="Run in dry-run mode without making actual changes"
)
args = parser.parse_args()
pyproject_path: str = "pyproject.toml"
# バージョン更新
new_version: Optional[str] = update_version(pyproject_path, args.dry_run)
if not new_version:
return # ユーザーが確認をキャンセルした場合、処理を中断
# uv sync 実行
run_uv_sync(args.dry_run)
# git 操作
git_operations(new_version, args.dry_run)
if __name__ == "__main__":
main()