#!/usr/bin/env python3 import requests import sys import datetime import socket # ================= KONFIGURÁCIÓ ================= INTERNAL_HOST = "gitea" EXTERNAL_IP = "192.168.100.10" PORT = "3000" OWNER = "kincses" REPO = "service-finder" TOKEN = "783f58519ee0ca060491dbc07f3dde1d8e48c5dd" HEADERS = { "Authorization": f"token {TOKEN}", "Content-Type": "application/json" } # Hibrid Hálózat-felismerés def get_base_url(): try: socket.gethostbyname(INTERNAL_HOST) return f"http://{INTERNAL_HOST}:{PORT}/api/v1" except socket.gaierror: return f"http://{EXTERNAL_IP}:{PORT}/api/v1" BASE_URL = get_base_url() LABELS = { "Status: To Do": "#ef4444", "Status: In Progress": "#f59e0b", "Status: Done": "#10b981", "Status: Blocked": "#000000", "Scope: Backend": "#0369a1", "Scope: Frontend": "#0284c7", "Scope: API": "#0ea5e9", "Scope: Core": "#38bdf8", "Scope: Robot": "#7dd3fc", "Scope: Database": "#ec4899", "Type: Script": "#8b5cf6", "Type: Model": "#3b82f6", "Type: Database": "#ec4899", "Type: Bug": "#dc2626", "Type: Feature": "#16a34a", "Type: Refactor": "#16a34a", "Role: Admin": "#fb923c", "Role: User": "#fdba74" } # ================================================ def fetch_all_pages(endpoint): """Gitea API lapozás (Pagination) kezelése, hogy minden elemet visszakapjunk.""" all_data = [] page = 1 limit = 50 separator = "&" if "?" in endpoint else "?" while True: url = f"{BASE_URL}{endpoint}{separator}limit={limit}&page={page}" res = requests.get(url, headers=HEADERS) if res.status_code != 200: break data = res.json() if not data: break all_data.extend(data) if len(data) < limit: break page += 1 return all_data def init_labels(): existing_labels = fetch_all_pages(f"/repos/{OWNER}/{REPO}/labels") existing = {l['name']: l['id'] for l in existing_labels} label_ids = {} for name, color in LABELS.items(): if name in existing: label_ids[name] = existing[name] else: post_res = requests.post(f"{BASE_URL}/repos/{OWNER}/{REPO}/labels", headers=HEADERS, json={"name": name, "color": color}) if post_res.status_code == 201: label_ids[name] = post_res.json()['id'] return label_ids def set_issue_state(issue_num, new_state_label, category_labels=[]): label_ids = init_labels() res = requests.get(f"{BASE_URL}/repos/{OWNER}/{REPO}/issues/{issue_num}/labels", headers=HEADERS) current_ids = [l['id'] for l in res.json()] if res.status_code == 200 else [] for status in ["Status: To Do", "Status: In Progress", "Status: Done", "Status: Blocked"]: if status in label_ids and label_ids[status] in current_ids: current_ids.remove(label_ids[status]) if new_state_label in label_ids and label_ids[new_state_label] not in current_ids: current_ids.append(label_ids[new_state_label]) for cat in category_labels: if cat in label_ids and label_ids[cat] not in current_ids: current_ids.append(label_ids[cat]) requests.put(f"{BASE_URL}/repos/{OWNER}/{REPO}/issues/{issue_num}/labels", headers=HEADERS, json={"labels": current_ids}) def add_comment(issue_num, message): requests.post(f"{BASE_URL}/repos/{OWNER}/{REPO}/issues/{issue_num}/comments", headers=HEADERS, json={"body": message}) # --- MÉRFÖLDKŐ (MILESTONE) KEZELÉS --- def resolve_milestone_id(name_or_id): if not name_or_id: return None if str(name_or_id).isdigit(): return int(name_or_id) milestones = fetch_all_pages(f"/repos/{OWNER}/{REPO}/milestones") for ms in milestones: if ms['title'].lower() == str(name_or_id).lower(): return ms['id'] return None def create_milestone(title, description="", due_date=None): existing_id = resolve_milestone_id(title) if existing_id: print(f"Mérföldkő már létezik: '{title}' (ID: {existing_id})") return existing_id payload = {"title": title, "description": description} if due_date: payload["due_on"] = f"{due_date}T23:59:59Z" if len(due_date) == 10 else due_date res = requests.post(f"{BASE_URL}/repos/{OWNER}/{REPO}/milestones", headers=HEADERS, json=payload) if res.status_code == 201: ms_id = res.json()['id'] print(f"✅ Mérföldkő sikeresen létrehozva: '{title}' (ID: {ms_id})") return ms_id print(f"❌ Hiba a mérföldkő létrehozásakor: {res.text}") return None def list_milestones(): milestones = fetch_all_pages(f"/repos/{OWNER}/{REPO}/milestones") print(f"\n{'ID':<5} | {'Mérföldkő Címe':<40} | {'Haladás'}") print("-" * 65) for ms in milestones: print(f"#{ms['id']:<4} | {ms['title'][:40]:<40} | {ms['completeness']}%") # --- KÁRTYA (ISSUE) KEZELÉS --- def create_issue(title, body, categories, milestone_ref=None, due_date=None, assignees=None): ms_id = resolve_milestone_id(milestone_ref) payload = {"title": title, "body": body} if ms_id: payload["milestone"] = ms_id if due_date: payload["due_date"] = f"{due_date}T23:59:59Z" if len(due_date) == 10 else due_date if assignees: payload["assignees"] = assignees res = requests.post(f"{BASE_URL}/repos/{OWNER}/{REPO}/issues", headers=HEADERS, json=payload) if res.status_code == 201: issue_num = res.json()['number'] set_issue_state(issue_num, "Status: To Do", categories) ms_text = f" (Milestone: {ms_id})" if ms_id else "" print(f"✅ Siker: #{issue_num} feladat létrehozva{ms_text}.") return True print(f"❌ Hiba a kártya létrehozásakor: {res.text}") return False def start_issue(issue_num): now = datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S") set_issue_state(issue_num, "Status: In Progress") requests.post(f"{BASE_URL}/repos/{OWNER}/{REPO}/issues/{issue_num}/stopwatch/start", headers=HEADERS) add_comment(issue_num, f"▶️ **Munka megkezdve:** {now}") print(f"✅ Siker: A #{issue_num} időmérése elindult.") def finish_issue(issue_num, custom_message=None): now = datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S") set_issue_state(issue_num, "Status: Done") requests.post(f"{BASE_URL}/repos/{OWNER}/{REPO}/issues/{issue_num}/stopwatch/stop", headers=HEADERS) requests.patch(f"{BASE_URL}/repos/{OWNER}/{REPO}/issues/{issue_num}", headers=HEADERS, json={"state": "closed"}) comment_body = f"✅ **Munka befejezve:** {now}\n\n**Technikai Összefoglaló:**\n{custom_message}\n\n⏱️ *A ráfordított időt a Gitea rögzítette.*" if custom_message else f"✅ **Munka befejezve:** {now}\n⏱️ *A ráfordított időt a Gitea rögzítette.*" add_comment(issue_num, comment_body) print(f"✅ Siker: A #{issue_num} lezárva, időmérés megállítva.") def get_issue(issue_num): res = requests.get(f"{BASE_URL}/repos/{OWNER}/{REPO}/issues/{issue_num}", headers=HEADERS) if res.status_code != 200: print(f"Hiba: Nem sikerült lekérni a #{issue_num} feladatot. Státusz: {res.status_code}") sys.exit(1) data = res.json() ms_title = data.get('milestone', {}).get('title', 'Nincs') if data.get('milestone') else 'Nincs' print("=" * 60) print(f"Feladat #{issue_num} - {data.get('state', 'unknown').upper()} (Mérföldkő: {ms_title})") print("=" * 60) print(f"Cím: {data.get('title', 'Nincs cím')}") print("-" * 60) print(data.get('body', 'Nincs leírás')) print("=" * 60) def list_issues(state="open"): issues = fetch_all_pages(f"/repos/{OWNER}/{REPO}/issues?state={state}") print(f"\n--- {state.upper()} FELADATOK ---") print(f"{'ID':<4} | {'Cím':<40} | {'Mérföldkő'}") print("-" * 70) for i in issues: ms = i.get('milestone', {}).get('title', '-') if i.get('milestone') else '-' print(f"#{i['number']:<3} | {i['title'][:40]:<40} | {ms}") # ================= FŐPROGRAM ================= if __name__ == "__main__": raw_args = sys.argv[1:] if not raw_args: print("Használat: python3 gitea_manager.py [parancs] [argumentumok]") print(" list - Nyitott kártyák listázása") print(" list closed - Lezárt kártyák listázása") print(" ms list - Mérföldkövek listázása") print(" ms create \"Név\" - Új mérföldkő létrehozása") print(" create \"Cím\" \"Leírás\" [Mérföldkő] [Címkék...] [--due YYYY-MM-DD] [--assign username]") print(" start - Munka megkezdése") print(" finish [msg] - Munka lezárása") print(" get - Kártya lekérése") sys.exit(1) # Paraméterek kinyerése (--due, --assign) args = [] due_date = None assignees = [] i = 0 while i < len(raw_args): if raw_args[i] == "--due" and i + 1 < len(raw_args): due_date = raw_args[i+1] i += 2 elif raw_args[i] == "--assign" and i + 1 < len(raw_args): assignees.append(raw_args[i+1]) i += 2 else: args.append(raw_args[i]) i += 1 action = args[0].lower() if action == "list": list_issues(args[1] if len(args) > 1 else "open") elif action == "ms": if len(args) > 1 and args[1].lower() == "create": create_milestone(args[2], args[3] if len(args) > 3 else "", due_date) else: list_milestones() elif action == "start" and len(args) > 1: start_issue(args[1]) elif action == "finish" and len(args) > 1: finish_issue(args[1], args[2] if len(args) > 2 else None) elif action == "get" and len(args) > 1: get_issue(args[1]) elif action == "create" and len(args) > 2: title, body = args[1], args[2] milestone_ref = None categories = [] if len(args) > 3: arg3 = args[3] if any(arg3.startswith(prefix) for prefix in ["Status:", "Scope:", "Type:", "Role:"]): categories = args[3:] else: milestone_ref = arg3 categories = args[4:] create_issue(title, body, categories, milestone_ref, due_date, assignees)