diff --git a/approved-tasks.json b/approved-tasks.json new file mode 100644 index 0000000..0acbce9 --- /dev/null +++ b/approved-tasks.json @@ -0,0 +1,4 @@ +{ + "projects": [], + "last_updated": null +} diff --git a/cron/jobs.json b/cron/jobs.json index 97ae695..b299874 100644 --- a/cron/jobs.json +++ b/cron/jobs.json @@ -179,12 +179,12 @@ }, { "name": "morning-report", - "cron": "30 9 * * *", + "cron": "30 8 * * *", "channel": "echo-work", "model": "sonnet", - "prompt": "RAPORT DIMINEAȚĂ - trimite pe EMAIL (Gmail: mmarius28@gmail.com)\n\n## CALENDAR\nVerifică calendarul:\n```bash\ncd ~/echo-core && source venv/bin/activate && python3 tools/calendar_check.py today\npython3 tools/calendar_check.py travel\npython3 tools/calendar_check.py week\n```\n\n## CITEȘTE CONTEXT\n- USER.md pentru programul lui Marius (luni-joi 15-16 liber)\n- memory/kb/insights/ pentru propuneri (ultimele 3 zile)\n- memory/approved-tasks.md pentru status proiecte/features\n\n## FORMAT EMAIL HTML\n- Font: 16px text, 18px titluri\n- Culori: albastru (#dbeafe) DONE, gri (#f3f4f6) PROGRAMAT, verde (#d1fae5) PROJECTS\n- Link-uri vizibile\n\n## STRUCTURA RAPORT\n\n### 1. CALENDAR\n- 📅 **AZI:** [evenimente]\n- 📅 **MÂINE:** [evenimente]\n- 📅 **PESTE 2 ZILE:** [dacă e GRUP, NLP, meeting mare]\n- 🚂 **TRAVEL:** Reminders bilete+cazare\n\n### 2. PROIECTE/FEATURES NOAPTEA 💻\n\nCitesc approved-tasks.md și raportez ce s-a realizat:\n\n**Format pentru fiecare proiect/feature [x]:**\n\n```html\n
\n

✅ P1 - Nume Proiect

\n \n

Status: X/Y stories complete

\n \n

Stories realizate:

\n \n \n

Link: gitea.romfast.ro/romfast/PROJECT-NAME

\n \n

Learnings: [din progress.txt - ce patterns am descoperit]

\n \n

Next steps: [ce rămâne de făcut]

\n
\n```\n\n**Dacă NU s-au executat proiecte/features:**\n- Sari peste această secțiune\n\n### 3. STATUS GENERAL\n- Ce s-a făcut ieri (joburi, taskuri)\n- Git status ~/clawd\n- Joburi executate (YouTube, insights, etc.)\n\n### 4. PROPUNERI CU ZI ȘI ORĂ!\n\n**OBLIGATORIU:** Fiecare propunere TU+EU sau FAC TU trebuie să aibă ZI și ORĂ concrete!\n\nCategorii:\n- 🤖 **FAC EU** (0 efort) - execut singur\n- 🤝 **TU+EU** (eu pregătesc) - cu zi/oră!\n- 👤 **FAC TU** (template gata) - cu zi/oră!\n\nExemplu:\n- **A1 - Sesiune Dizolvare Vină** 🤝 TU+EU\n 📅 **Marți 3 feb, 15:00-15:30**\n Context + link sursă\n\nReguli programare:\n- Luni-Joi 15:00-16:00 = slot liber\n- Vineri-Duminică = NLP, evită\n- Verifică calendar să nu fie ocupat\n\n### 5. INSIGHTS DISPONIBILE\n\nListează insights-uri [ ] nepropuse încă (format scurt).\n\n### 6. CUM RĂSPUNZI\n- DA = aprob toate (cu zilele/orele propuse)\n- 1 pentru A1,A2 = execut ACUM\n- 2 pentru A3 = programez noapte\n- 3 pentru A5 = skip\n- Alt orar = \"A1 miercuri nu marți\"\n\n## TRIMITERE\npython3 /home/moltbot/echo-core/tools/email_send.py \"mmarius28@gmail.com\" \"Raport Dimineata DATA\" \"HTML_CONTENT\"\n\nNU trimite pe Discord - doar email.", + "enabled": true, + "prompt": "RAPORT DIMINEAȚĂ - trimite pe EMAIL (Gmail: mmarius28@gmail.com)\n\n## CALENDAR\nVerifică calendarul:\n```bash\ncd ~/echo-core && source venv/bin/activate && python3 tools/calendar_check.py today\npython3 tools/calendar_check.py travel\npython3 tools/calendar_check.py week\n```\n\n## CITEȘTE CONTEXT\n- USER.md pentru programul lui Marius (luni-joi 15-16 liber)\n- memory/kb/insights/ pentru propuneri (ultimele 3 zile)\n- /home/moltbot/echo-core/approved-tasks.json pentru status proiecte/features (câmpurile: name, status, started_at, pid)\n\n## FORMAT EMAIL HTML\n- Font: 16px text, 18px titluri\n- Culori: albastru (#dbeafe) DONE, gri (#f3f4f6) PROGRAMAT, verde (#d1fae5) PROJECTS\n- Link-uri vizibile\n\n## STRUCTURA RAPORT\n\n### 1. CALENDAR\n- 📅 **AZI:** [evenimente]\n- 📅 **MÂINE:** [evenimente]\n- 📅 **PESTE 2 ZILE:** [dacă e GRUP, NLP, meeting mare]\n- 🚂 **TRAVEL:** Reminders bilete+cazare\n\n### 2. PROIECTE/FEATURES NOAPTEA 💻\n\nCitesc /home/moltbot/echo-core/approved-tasks.json și raportez ce s-a realizat:\n(statusuri: pending, approved, running, complete, failed, stopped)\nPentru stories done/total: citesc /home/moltbot/workspace/{name}/scripts/ralph/prd.json\n\n**Format pentru fiecare proiect/feature [x]:**\n\n```html\n
\n

✅ P1 - Nume Proiect

\n \n

Status: X/Y stories complete

\n \n

Stories realizate:

\n \n \n

Link: gitea.romfast.ro/romfast/PROJECT-NAME

\n \n

Learnings: [din progress.txt - ce patterns am descoperit]

\n \n

Next steps: [ce rămâne de făcut]

\n
\n```\n\n**Dacă NU s-au executat proiecte/features:**\n- Sari peste această secțiune\n\n### 3. STATUS GENERAL\n- Ce s-a făcut ieri (joburi, taskuri)\n- Git status ~/clawd\n- Joburi executate (YouTube, insights, etc.)\n\n### 4. PROPUNERI CU ZI ȘI ORĂ!\n\n**OBLIGATORIU:** Fiecare propunere TU+EU sau FAC TU trebuie să aibă ZI și ORĂ concrete!\n\nCategorii:\n- 🤖 **FAC EU** (0 efort) - execut singur\n- 🤝 **TU+EU** (eu pregătesc) - cu zi/oră!\n- 👤 **FAC TU** (template gata) - cu zi/oră!\n\nExemplu:\n- **A1 - Sesiune Dizolvare Vină** 🤝 TU+EU\n 📅 **Marți 3 feb, 15:00-15:30**\n Context + link sursă\n\nReguli programare:\n- Luni-Joi 15:00-16:00 = slot liber\n- Vineri-Duminică = NLP, evită\n- Verifică calendar să nu fie ocupat\n\n### 5. INSIGHTS DISPONIBILE\n\nListează insights-uri [ ] nepropuse încă (format scurt).\n\n### 6. CUM RĂSPUNZI\n- DA = aprob toate (cu zilele/orele propuse)\n- 1 pentru A1,A2 = execut ACUM\n- 2 pentru A3 = programez noapte\n- 3 pentru A5 = skip\n- Alt orar = \"A1 miercuri nu marți\"\n\n## TRIMITERE\npython3 /home/moltbot/echo-core/tools/email_send.py \"mmarius28@gmail.com\" \"Raport Dimineata DATA\" \"HTML_CONTENT\"\n\nNU trimite pe Discord - doar email.", "allowed_tools": [], - "enabled": false, "last_run": null, "last_status": null, "next_run": null @@ -194,9 +194,9 @@ "cron": "0 21 * * *", "channel": "echo-work", "model": "sonnet", - "prompt": "RAPORT SEARĂ - trimite pe EMAIL (Gmail: mmarius28@gmail.com)\n\n## CALENDAR\nVerifică ce ai mâine și săptămâna:\n```bash\ncd ~/echo-core && source venv/bin/activate && python3 tools/calendar_check.py today\npython3 tools/calendar_check.py week\n```\n\n## CITEȘTE CONTEXT\n- USER.md pentru programul lui Marius (luni-joi 15-16 liber, vineri-dum NLP)\n- memory/kb/insights/YYYY-MM-DD.md pentru propuneri insights\n- memory/kb/youtube/ și memory/kb/articole/ pentru inspirație proiecte\n- memory/approved-tasks.md pentru status proiecte existente\n\n## FORMAT EMAIL HTML\n- Font: 16px text, 18px titluri\n- Culori: albastru (#dbeafe) DONE, gri (#f3f4f6) PROGRAMAT, verde (#d1fae5) PROJECTS\n- Link-uri vizibile\n\n## STRUCTURA RAPORT\n\n### 1. MÂINE\n- 📅 Evenimente calendar\n- 🚂 Travel reminders\n\n### 2. STATUS\n- Ce s-a făcut azi\n- Git status\n\n### 3. PROPUNERI CU ZI ȘI ORĂ!\n\n**OBLIGATORIU:** Fiecare propunere TU+EU sau FAC TU trebuie să aibă ZI și ORĂ concrete!\n\nReguli programare:\n- Luni-Joi 15:00-16:00 = slot liber\n- Vineri-Duminică = NLP, evită\n- Verifică calendar să nu fie ocupat\n- Sesiuni scurte: 15-30 min\n\n### 4. PROGRAME/PROIECTE PRACTICE 💻\n\n**CONTEXT OBLIGATORIU - citește înainte de a propune:**\n\n**Proiecte existente (PRIORITARE pentru features):**\n- **roa2web** (gitea.romfast.ro/romfast/roa2web) - FastAPI+Vue.js+Telegram bot\n - Are deja: balanță, facturi, trezorerie\n - Lipsesc: validări declarații ANAF, facturare valută/taxare inversă, notificări\n - Rapoarte ROA noi → FEATURE în roa2web, NU proiect separat!\n- **Chatbot Maria** (Flowise pe LXC 104, ngrok → romfast.ro/chatbot_maria.html)\n - Document store: XML, MD | Groq gratuit + Ollama embeddings + FAISS\n - Problema: răspunsuri nu sunt suficient de bune\n - Angajatul nou poate menține documentația (scrie TXT, trebuie converter)\n - Clientii îl accesează din programele ROA direct\n\n**Întrebări frecvente clienți (surse de proiecte):**\n- Erori validare declarații ANAF (D406, D394, D100 etc.)\n- Cum facturez în valută cu taxare inversă?\n- Probleme la instalări, inițializări firme noi, configurări\n\n**Reguli propuneri (80/20 STRICT):**\n- Impact mare pentru Marius → apoi pentru clienți ERP ROA\n- Inspirat din discovery (YouTube, articole, insights procesate)\n- Features roa2web > proiecte noi (integrare în existent)\n- Proiecte independente doar dacă NU se potrivesc în roa2web/Flowise\n\n**A. FEATURES PROIECTE EXISTENTE (2-3, PRIORITAR):**\n\nFormat:\n```\n### ⚡ F1 - Feature pentru [roa2web/chatbot]\n**Ce face:** Descriere scurtă\n**De ce:** Ce problemă rezolvă (ex: \"clienții întreabă X de 5 ori/săptămână\")\n**Complexitate:** S/M/L\n**Proiect:** roa2web / chatbot-maria\n```\n\n**B. PROIECTE NOI (max 1, doar dacă nu se integrează în existente):**\n\nFormat:\n```\n### 💻 P1 - Nume Proiect\n**De ce:** Cum se leagă de nevoile lui Marius/clienți\n**Impact:** Pentru Marius + pentru clienți\n**Efort:** Ore/zile realist\n**Stack:** Simplu (80/20)\n**Sursă:** [Link nota KB]\n```\n\n**NU propune:**\n- Proiecte complexe fără beneficiu clar\n- Proiecte duplicat cu ce există deja\n- Rapoarte ROA ca proiect separat (→ feature roa2web)\n\n### 5. INSIGHTS DISPONIBILE\nListează insights-uri [ ] nepropuse încă (format scurt).\n\n### 6. CUM RĂSPUNZI\n- DA = aprob toate (cu zilele/orele propuse)\n- 1 pentru A1,A2 = execut ACUM\n- 2 pentru A3 = programez noapte\n- 3 pentru A5 = skip\n- **F pentru F1,F3** = implementează features (joburi noapte)\n- **P pentru P1** = creează proiect nou (job noapte)\n- Alt orar = \"A1 miercuri nu marți\"\n\n## IMPLEMENTARE PROIECTE APROBATE\n\nCând Marius aprobă cu F sau P:\n1. Adaugă în memory/approved-tasks.md secțiunea \"Noaptea asta\"\n2. Night-execute (23:00) va:\n - Pentru proiecte noi: genera PRD cu ralph_prd_generator.py → ralph.sh\n - Pentru features roa2web: adaugă stories în prd.json existent → ralph.sh\n - Pentru chatbot: generează documentație nouă și uploadează în Flowise\n\n## TRIMITERE\npython3 /home/moltbot/echo-core/tools/email_send.py \"mmarius28@gmail.com\" \"Raport Seara DATA\" \"HTML_CONTENT\"\n\nNU trimite pe Discord - doar email.", + "enabled": true, + "prompt": "RAPORT SEARĂ - trimite pe EMAIL (Gmail: mmarius28@gmail.com)\n\n## CALENDAR\nVerifică ce ai mâine și săptămâna:\n```bash\ncd ~/echo-core && source venv/bin/activate && python3 tools/calendar_check.py today\npython3 tools/calendar_check.py week\n```\n\n## CITEȘTE CONTEXT\n- USER.md pentru programul lui Marius (luni-joi 15-16 liber, vineri-dum NLP)\n- memory/kb/insights/YYYY-MM-DD.md pentru propuneri insights\n- memory/kb/youtube/ și memory/kb/articole/ pentru inspirație proiecte\n- /home/moltbot/echo-core/approved-tasks.json pentru status proiecte existente (câmpurile: name, status, proposed_at)\n\n## FORMAT EMAIL HTML\n- Font: 16px text, 18px titluri\n- Culori: albastru (#dbeafe) DONE, gri (#f3f4f6) PROGRAMAT, verde (#d1fae5) PROJECTS\n- Link-uri vizibile\n\n## STRUCTURA RAPORT\n\n### 1. MÂINE\n- 📅 Evenimente calendar\n- 🚂 Travel reminders\n\n### 2. STATUS\n- Ce s-a făcut azi\n- Git status\n\n### 3. PROPUNERI CU ZI ȘI ORĂ!\n\n**OBLIGATORIU:** Fiecare propunere TU+EU sau FAC TU trebuie să aibă ZI și ORĂ concrete!\n\nReguli programare:\n- Luni-Joi 15:00-16:00 = slot liber\n- Vineri-Duminică = NLP, evită\n- Verifică calendar să nu fie ocupat\n- Sesiuni scurte: 15-30 min\n\n### 4. PROGRAME/PROIECTE PRACTICE 💻\n\n**CONTEXT OBLIGATORIU - citește înainte de a propune:**\n\n**Proiecte existente (PRIORITARE pentru features):**\n- **roa2web** (gitea.romfast.ro/romfast/roa2web) - FastAPI+Vue.js+Telegram bot\n - Are deja: balanță, facturi, trezorerie\n - Lipsesc: validări declarații ANAF, facturare valută/taxare inversă, notificări\n - Rapoarte ROA noi → FEATURE în roa2web, NU proiect separat!\n- **Chatbot Maria** (Flowise pe LXC 104, ngrok → romfast.ro/chatbot_maria.html)\n - Document store: XML, MD | Groq gratuit + Ollama embeddings + FAISS\n - Problema: răspunsuri nu sunt suficient de bune\n - Angajatul nou poate menține documentația (scrie TXT, trebuie converter)\n - Clientii îl accesează din programele ROA direct\n\n**Întrebări frecvente clienți (surse de proiecte):**\n- Erori validare declarații ANAF (D406, D394, D100 etc.)\n- Cum facturez în valută cu taxare inversă?\n- Probleme la instalări, inițializări firme noi, configurări\n\n**Reguli propuneri (80/20 STRICT):**\n- Impact mare pentru Marius → apoi pentru clienți ERP ROA\n- Inspirat din discovery (YouTube, articole, insights procesate)\n- Features roa2web > proiecte noi (integrare în existent)\n- Proiecte independente doar dacă NU se potrivesc în roa2web/Flowise\n\n**A. FEATURES PROIECTE EXISTENTE (2-3, PRIORITAR):**\n\nFormat:\n```\n### ⚡ F1 - Feature pentru [roa2web/chatbot]\n**Ce face:** Descriere scurtă\n**De ce:** Ce problemă rezolvă (ex: \"clienții întreabă X de 5 ori/săptămână\")\n**Complexitate:** S/M/L\n**Proiect:** roa2web / chatbot-maria\n```\n\n**B. PROIECTE NOI (max 1, doar dacă nu se integrează în existente):**\n\nFormat:\n```\n### 💻 P1 - Nume Proiect\n**De ce:** Cum se leagă de nevoile lui Marius/clienți\n**Impact:** Pentru Marius + pentru clienți\n**Efort:** Ore/zile realist\n**Stack:** Simplu (80/20)\n**Sursă:** [Link nota KB]\n```\n\n**NU propune:**\n- Proiecte complexe fără beneficiu clar\n- Proiecte duplicat cu ce există deja\n- Rapoarte ROA ca proiect separat (→ feature roa2web)\n\n### 5. INSIGHTS DISPONIBILE\nListează insights-uri [ ] nepropuse încă (format scurt).\n\n### 6. CUM RĂSPUNZI\n- DA = aprob toate (cu zilele/orele propuse)\n- 1 pentru A1,A2 = execut ACUM\n- 2 pentru A3 = programez noapte\n- 3 pentru A5 = skip\n- **F pentru F1,F3** = implementează features (joburi noapte)\n- **P pentru P1** = creează proiect nou (job noapte)\n- Alt orar = \"A1 miercuri nu marți\"\n\n## IMPLEMENTARE PROIECTE APROBATE\n\nCând propui features (F) sau proiecte (P), adaugă-le automat în /home/moltbot/echo-core/approved-tasks.json cu status 'pending':\n```bash\npython3 -c \"\nimport json, datetime\nf = open('/home/moltbot/echo-core/approved-tasks.json')\ndata = json.load(f); f.close()\ndata['projects'].append({'name': 'SLUG-PROIECT', 'description': 'DESCRIERE', 'status': 'pending', 'proposed_at': datetime.datetime.utcnow().isoformat(), 'approved_at': None, 'started_at': None, 'pid': None})\ndata['last_updated'] = datetime.datetime.utcnow().isoformat()\nopen('/home/moltbot/echo-core/approved-tasks.json', 'w').write(json.dumps(data, indent=2))\n\"\n```\n\nÎn email, arată lui Marius comanda de aprobare:\n`!approve SLUG-PROIECT` (trimite pe Discord/Telegram la Echo)\n\nNight-execute (23:00) va:\n - genera PRD cu ralph_prd_generator.py dacă nu există prd.json\n - lansa ralph.sh 15 iterații pentru fiecare proiect aprobat\n\n## TRIMITERE\npython3 /home/moltbot/echo-core/tools/email_send.py \"mmarius28@gmail.com\" \"Raport Seara DATA\" \"HTML_CONTENT\"\n\nNU trimite pe Discord - doar email.", "allowed_tools": [], - "enabled": false, "last_run": null, "last_status": null, "next_run": null @@ -269,8 +269,24 @@ "prompt": "Heartbeat check. Rulează src/heartbeat.py printr-un scurt raport de status.\nDacă nu e nimic de raportat (email=0, calendar nu are evenimente <2h, kb ok), răspunde doar cu HEARTBEAT_OK și oprește-te — nu trimite mesaj.\nDacă e ceva: raport scurt pe Discord #echo-work.", "allowed_tools": [], "enabled": true, - "last_run": "2026-04-26T08:00:00.003149+00:00", + "last_run": "2026-04-26T14:00:00.003075+00:00", "last_status": "ok", - "next_run": "2026-04-26T10:00:00+00:00" + "next_run": "2026-04-26T16:00:00+00:00" + }, + { + "name": "night-execute", + "cron": "0 23 * * *", + "channel": "echo-work", + "model": "opus", + "enabled": true, + "prompt": "NIGHT-EXECUTE - Implementare autonoma proiecte aprobate\n\n## PASUL 1: Citeste proiectele aprobate\n\nCiteste /home/moltbot/echo-core/approved-tasks.json\nSelecteaza proiectele cu status='approved'\nDaca nu sunt proiecte aprobate: raporteaza pe Discord si opreste-te.\n\n## PASUL 2: Pentru fiecare proiect aprobat\n\n1. Verifica daca workspace-ul exista: /home/moltbot/workspace/{name}\n - Daca nu: git clone https://gitea.romfast.ro/romfast/{name}.git /home/moltbot/workspace/{name}\n\n2. Verifica daca prd.json exista: /home/moltbot/workspace/{name}/scripts/ralph/prd.json\n - Daca nu: ruleaza generatorul PRD:\n source .venv/bin/activate\n python3 tools/ralph_prd_generator.py \"{name}\" \"{description}\" /home/moltbot/workspace\n\n3. Lanseaza Ralph loop:\n cd /home/moltbot/workspace/{name}\n chmod +x scripts/ralph/ralph.sh\n mkdir -p scripts/ralph/logs\n nohup ./scripts/ralph/ralph.sh 15 > scripts/ralph/logs/ralph-$(date +%Y%m%d).log 2>&1 &\n echo $! > scripts/ralph/.ralph.pid\n\n4. Actualizeaza approved-tasks.json:\n - status: 'running'\n - started_at: timestamp curent\n - pid: PID din .ralph.pid\n\n## PASUL 3: Raport Discord\n\nTrimite pe echo-work:\n- Cate proiecte au pornit\n- PID-urile lor\n- 'morning-report va raporta progresul la 08:30'\n\n## REGULI IMPORTANTE\n\n- Nu modifica niciodata src/router.py, src/claude_session.py sau alte fisiere core echo-core prin Ralph\n- echo-core self-improvement NUMAI pe branch ralph/echo-improve, nu pe master\n- Daca ralph.sh esueaza: log in approved-tasks.json (status: failed, error: mesaj)\n- Delay 5 secunde intre proiecte pentru a evita rate limiting\n", + "allowed_tools": [ + "Bash", + "Read", + "Write" + ], + "last_run": null, + "last_status": null, + "next_run": null } -] +] \ No newline at end of file diff --git a/src/router.py b/src/router.py index 3311735..8aa434d 100644 --- a/src/router.py +++ b/src/router.py @@ -1,6 +1,11 @@ """Echo Core message router — routes messages to Claude or commands.""" +import json import logging +import os +import signal +from datetime import datetime, timezone +from pathlib import Path from typing import Callable from src.config import Config @@ -16,6 +21,8 @@ from src.claude_session import ( log = logging.getLogger(__name__) +APPROVED_TASKS_FILE = Path(__file__).parent.parent / "approved-tasks.json" + # Module-level config instance (lazy singleton) _config: Config | None = None @@ -45,6 +52,19 @@ def route_message( """ text = text.strip() + # Ralph commands (!approve, !status, !stop, !propose) + if text.lower().startswith("!approve ") or text.lower() == "!approve": + return _ralph_approve(text), True + + if text.lower() == "!status" or text.lower().startswith("!status "): + return _ralph_status(text), True + + if text.lower().startswith("!stop "): + return _ralph_stop(text), True + + if text.lower().startswith("!propose "): + return _ralph_propose(text), True + # Text-based commands (not slash commands — these work in any adapter) if text.lower() == "/clear": default_model = _get_config().get("bot.default_model", "sonnet") @@ -136,6 +156,168 @@ def _model_command(channel_id: str, text: str) -> str: return f"Model changed to {choice}." +def _load_approved_tasks() -> dict: + """Load approved-tasks.json, return empty structure if missing.""" + if APPROVED_TASKS_FILE.exists(): + return json.loads(APPROVED_TASKS_FILE.read_text()) + return {"projects": [], "last_updated": None} + + +def _save_approved_tasks(data: dict) -> None: + data["last_updated"] = datetime.now(timezone.utc).isoformat() + APPROVED_TASKS_FILE.write_text(json.dumps(data, indent=2, ensure_ascii=False)) + + +def _ralph_approve(text: str) -> str: + """!approve P1,P2 sau !approve roa2web — aprobă proiecte pentru night-execute.""" + parts = text.split(None, 1) + if len(parts) < 2: + data = _load_approved_tasks() + pending = [p for p in data["projects"] if p.get("status") == "pending"] + if not pending: + return "Niciun proiect pending. Folosește !propose pentru a adăuga." + lines = [f"Proiecte pending (aprobă cu !approve ):"] + for p in pending: + lines.append(f" - {p['name']}: {p['description'][:60]}") + return "\n".join(lines) + + names_raw = parts[1].strip() + names = [n.strip() for n in names_raw.replace(",", " ").split() if n.strip()] + + data = _load_approved_tasks() + approved = [] + not_found = [] + + for name in names: + found = False + for p in data["projects"]: + if p["name"].lower() == name.lower(): + p["status"] = "approved" + p["approved_at"] = datetime.now(timezone.utc).isoformat() + approved.append(p["name"]) + found = True + break + if not found: + not_found.append(name) + + if not_found: + return f"Nu am găsit proiectele: {', '.join(not_found)}. Verifică !status pentru lista completă." + + _save_approved_tasks(data) + names_str = ", ".join(approved) + return f"✅ Aprobat pentru tonight: {names_str}\nNight-execute rulează la 23:00 și va implementa stories autonom." + + +def _ralph_status(text: str) -> str: + """!status sau !status — status Ralph pentru proiecte.""" + parts = text.split(None, 1) + filter_name = parts[1].strip().lower() if len(parts) > 1 else None + + data = _load_approved_tasks() + projects = data.get("projects", []) + + if filter_name: + projects = [p for p in projects if filter_name in p["name"].lower()] + + if not projects: + return "Niciun proiect în approved-tasks.json. Adaugă cu !propose." + + lines = ["📊 Status proiecte Ralph:"] + for p in projects: + status = p.get("status", "unknown") + name = p["name"] + pid = p.get("pid") + started = p.get("started_at", "")[:16] if p.get("started_at") else "-" + + # Verifică dacă procesul mai rulează + if pid and status == "running": + try: + os.kill(pid, 0) + running_indicator = f"🟢 PID {pid}" + except (ProcessLookupError, PermissionError): + running_indicator = "🔴 PID mort" + p["status"] = "stopped" + _save_approved_tasks(data) + else: + running_indicator = {"approved": "⏳ aștept 23:00", "pending": "📋 pending", + "complete": "✅ complet", "failed": "❌ eșuat", "stopped": "⏹ oprit"}.get(status, status) + + # Stories complete din prd.json + prd_path = Path(f"/home/moltbot/workspace/{name}/scripts/ralph/prd.json") + stories_info = "" + if prd_path.exists(): + try: + prd = json.loads(prd_path.read_text()) + total = len(prd.get("userStories", [])) + done = sum(1 for s in prd.get("userStories", []) if s.get("passes")) + stories_info = f" | Stories: {done}/{total}" + except Exception: + pass + + lines.append(f" {name}: {running_indicator}{stories_info} | Start: {started}") + + return "\n".join(lines) + + +def _ralph_stop(text: str) -> str: + """!stop — oprește Ralph loop pentru un proiect.""" + parts = text.split(None, 1) + if len(parts) < 2: + return "Folosire: !stop " + + name = parts[1].strip() + data = _load_approved_tasks() + + for p in data["projects"]: + if p["name"].lower() == name.lower(): + pid = p.get("pid") + if pid: + try: + os.kill(pid, signal.SIGTERM) + p["status"] = "stopped" + p["stopped_at"] = datetime.now(timezone.utc).isoformat() + _save_approved_tasks(data) + return f"⏹ Ralph oprit pentru {name} (PID {pid} terminat)." + except ProcessLookupError: + p["status"] = "stopped" + _save_approved_tasks(data) + return f"PID {pid} nu mai rula pentru {name}. Status actualizat." + except PermissionError: + return f"❌ Nu am permisiune să opresc PID {pid}." + else: + return f"{name} nu are un PID activ (status: {p.get('status', 'unknown')})." + + return f"Proiect '{name}' nu găsit în approved-tasks.json." + + +def _ralph_propose(text: str) -> str: + """!propose — adaugă un proiect pentru aprobare.""" + parts = text.split(None, 2) + if len(parts) < 3: + return "Folosire: !propose \nEx: !propose roa2web Homepage redesign cu hero section și animații" + + name = parts[1].strip() + description = parts[2].strip() + + data = _load_approved_tasks() + + for p in data["projects"]: + if p["name"].lower() == name.lower(): + return f"Proiectul '{name}' există deja cu status: {p.get('status', 'unknown')}." + + data["projects"].append({ + "name": name, + "description": description, + "status": "pending", + "proposed_at": datetime.now(timezone.utc).isoformat(), + "approved_at": None, + "started_at": None, + "pid": None + }) + _save_approved_tasks(data) + return f"📋 Adăugat: {name}\n{description}\n\nAprobă cu: !approve {name}" + + def _get_channel_config(channel_id: str) -> dict | None: """Find channel config by ID.""" channels = _get_config().get("channels", {}) diff --git a/tools/ralph/prd-template.json b/tools/ralph/prd-template.json new file mode 100644 index 0000000..42637e9 --- /dev/null +++ b/tools/ralph/prd-template.json @@ -0,0 +1,19 @@ +{ + "projectName": "feature-name", + "branchName": "ralph/feature-name", + "description": "Descriere scurtă a feature-ului", + "userStories": [ + { + "id": "US-001", + "title": "Titlu story", + "description": "As a [user type], I want [feature] so that [benefit]", + "priority": 1, + "acceptanceCriteria": [ + "Criteriu specific și verificabil", + "npm run typecheck passes" + ], + "passes": false, + "notes": "" + } + ] +} diff --git a/tools/ralph/prompt.md b/tools/ralph/prompt.md new file mode 100644 index 0000000..25f7f9b --- /dev/null +++ b/tools/ralph/prompt.md @@ -0,0 +1,207 @@ +# Ralph - Instrucțiuni pentru Iterație + +Ești un agent autonom care implementează user stories dintr-un PRD. Aceasta este O SINGURĂ iterație - implementezi UN singur story și apoi te oprești. + +## Workflow pentru această iterație + +### 1. Citește contextul +- PRD-ul și progress.txt sunt furnizate în context +- Înțelege ce stories sunt deja complete (`passes: true`) +- Identifică următorul story de implementat (prioritate cea mai mică dintre cele incomplete) +- Notează `techStack.commands` din PRD pentru comenzile corecte + +### 2. Management branch +- Verifică dacă ești pe branch-ul corect (specificat în `branchName` din PRD) +- Dacă nu, creează și checkout branch-ul: + ```bash + git checkout -b + ``` +- Dacă branch-ul există deja, doar checkout: + ```bash + git checkout + ``` + +### 3. Selectează story-ul +- Alege story-ul cu cea mai mică prioritate care are `passes: false` +- Citește atent acceptance criteria +- Verifică câmpul `requiresBrowserCheck` - dacă e `true`, trebuie verificare vizuală + +### 4. Implementare +- Implementează DOAR acest story +- Urmează patterns existente în codebase +- Fii minimal și focusat - nu adăuga funcționalități extra + +### 5. Quality Checks +Rulează TOATE verificările înainte de commit. Folosește comenzile din `techStack.commands`: + +```bash +# Folosește comenzile din prd.json techStack.commands: +{techStack.commands.typecheck} # Type checking +{techStack.commands.lint} # Linting +{techStack.commands.test} # Tests (dacă există) +``` + +**Comenzi standard per stack:** + +| Stack | Typecheck | Lint | Test | +|-------|-----------|------|------| +| Next.js/TS | npm run typecheck | npm run lint | npm test | +| Node.js | npm run typecheck | npm run lint | npm test | +| Python | mypy . | ruff check . | python -m pytest | +| Go | - | golangci-lint run | go test ./... | + +**IMPORTANT**: Nu face commit dacă verificările eșuează. Repară mai întâi. + +### 6. Verificare Browser (pentru UI stories) + +**DACĂ story-ul are `requiresBrowserCheck: true` sau implică UI:** + +Folosește **agent-browser CLI** pentru verificare vizuală. Agent-browser e optimizat pentru agenți AI cu referințe compacte (@e1, @e2) care consumă minim tokeni. + +#### 6.1 Pornește dev server-ul +```bash +# Folosește comanda din techStack.commands.start +{techStack.commands.start} +# Exemplu: npm run dev +``` + +Așteaptă să pornească (verifică output-ul pentru "ready" sau similar). + +#### 6.2 Navighează la pagină +```bash +agent-browser navigate "http://localhost:{techStack.port}" +# Exemplu: agent-browser navigate "http://localhost:3000" +``` + +#### 6.3 Ia snapshot pentru verificare +```bash +agent-browser snapshot +``` + +Snapshot-ul returnează o listă de elemente cu referințe compacte: +``` +@e1: heading "Welcome" +@e2: button "Login" +@e3: textbox "Email" +@e4: textbox "Password" +@e5: button "Submit" +``` + +**Verifică în snapshot:** +- Elementele cheie din acceptance criteria există +- Textul e corect +- Structura paginii e corectă + +#### 6.4 Testează interacțiunile (dacă e cazul) +```bash +# Click pe un element +agent-browser click @e2 + +# Fill un input +agent-browser fill @e3 "test@example.com" + +# Așteaptă o schimbare +agent-browser snapshot # verifică noua stare +``` + +#### 6.5 Salvează screenshot ca dovadă +```bash +agent-browser screenshot ./scripts/ralph/screenshots/US-{id}-$(date +%Y%m%d-%H%M%S).png +# Exemplu: agent-browser screenshot ./scripts/ralph/screenshots/US-001-20240115-143022.png +``` + +#### 6.6 Verifică erori +```bash +# Verifică console pentru erori +agent-browser console +``` + +**IMPORTANT**: +- NU marca story-ul complete dacă verificarea vizuală eșuează! +- Dacă găsești erori în browser, repară-le înainte de commit +- Screenshots sunt salvate în `scripts/ralph/screenshots/` pentru referință + +### 7. Documentare (dacă ai descoperit ceva util) +Dacă ai descoperit patterns sau gotchas, actualizează `AGENTS.md` în directorul relevant: +- API patterns +- Dependențe non-evidente +- Convenții de cod +- Cum să testezi anumite funcționalități + +### 8. Commit +Format commit message: +``` +feat: [Story ID] - [Story Title] +``` + +### 9. Marchează story-ul ca complet +**CRITIC**: Actualizează `scripts/ralph/prd.json`: +- Setează `passes: true` pentru story-ul implementat +- Adaugă note relevante în câmpul `notes` + +### 10. Actualizează progress.txt +Adaugă la sfârșitul fișierului `scripts/ralph/progress.txt`: + +```markdown +## Iterație: [timestamp] +### Story implementat: [ID] - [Title] +### Status: Complete + +### Verificări: +- Typecheck: PASS +- Lint: PASS +- Tests: PASS/SKIP +- Browser check: PASS/N/A + +### Learnings: +- [Ce ai învățat] +- [Patterns descoperite] + +### Next steps: +- [Ce rămâne de făcut] +--- +``` + +## Reguli importante + +1. **UN SINGUR STORY PE ITERAȚIE** - Nu implementa mai mult de un story +2. **TOATE CHECKS TREBUIE SĂ TREACĂ** - Nu face commit cu erori +3. **VERIFICARE BROWSER PENTRU UI** - Obligatorie dacă `requiresBrowserCheck: true` +4. **ACTUALIZEAZĂ prd.json** - Altfel iterația următoare va repeta munca +5. **FII CONCIS** - Nu over-engineer + +## Comenzi agent-browser (referință rapidă) + +```bash +# Navigare +agent-browser navigate "http://localhost:3000/page" + +# Snapshot (vedere compactă a paginii) +agent-browser snapshot + +# Click pe element (folosind ref din snapshot) +agent-browser click @e5 + +# Fill input +agent-browser fill @e3 "value" + +# Screenshot +agent-browser screenshot ./path/to/file.png + +# Console logs +agent-browser console + +# Așteaptă text +agent-browser wait-for "Loading complete" +``` + +## Condiție de terminare + +Dacă TOATE stories au `passes: true`, răspunde cu: + +``` +COMPLETE +``` + +--- +ÎNCEPE IMPLEMENTAREA ACUM. diff --git a/tools/ralph/ralph.sh b/tools/ralph/ralph.sh new file mode 100755 index 0000000..0ab755e --- /dev/null +++ b/tools/ralph/ralph.sh @@ -0,0 +1,229 @@ +#!/bin/bash +# Ralph pentru Claude Code - Loop autonom de agent AI +# Adaptat din Ralph original (snarktank/ralph) pentru Claude Code CLI +# Usage: ./ralph.sh [max_iterations] [project_dir] + +set -e + +MAX_ITERATIONS=${1:-10} +PROJECT_DIR=${2:-$(pwd)} +SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)" +PRD_FILE="$SCRIPT_DIR/prd.json" +PROGRESS_FILE="$SCRIPT_DIR/progress.txt" +ARCHIVE_DIR="$SCRIPT_DIR/archive" +SCREENSHOTS_DIR="$SCRIPT_DIR/screenshots" +LAST_BRANCH_FILE="$SCRIPT_DIR/.last-branch" +PROMPT_FILE="$SCRIPT_DIR/prompt.md" + +# Verifică că jq este instalat +if ! command -v jq &> /dev/null; then + echo "Eroare: jq nu este instalat. Rulează: apt install jq" + exit 1 +fi + +# Verifică că claude este instalat +if ! command -v claude &> /dev/null; then + echo "Eroare: Claude Code CLI nu este instalat." + echo "Instalează cu: npm install -g @anthropic-ai/claude-code" + exit 1 +fi + +# Verifică agent-browser (opțional, pentru verificări UI) +if ! command -v agent-browser &> /dev/null; then + echo "Notă: agent-browser nu este instalat." + echo "Pentru verificări vizuale UI, instalează cu: npm install -g agent-browser && agent-browser install" + echo "Continuăm fără verificări browser..." + echo "" +fi + +# Verifică existența fișierelor necesare +if [ ! -f "$PRD_FILE" ]; then + echo "Eroare: prd.json nu există în $SCRIPT_DIR" + echo "Generează mai întâi un PRD folosind skill-ul /prd și apoi /ralph" + exit 1 +fi + +if [ ! -f "$PROMPT_FILE" ]; then + echo "Eroare: prompt.md nu există în $SCRIPT_DIR" + exit 1 +fi + +# Arhivare rulare anterioară dacă branch-ul s-a schimbat +if [ -f "$PRD_FILE" ] && [ -f "$LAST_BRANCH_FILE" ]; then + CURRENT_BRANCH=$(jq -r '.branchName // empty' "$PRD_FILE" 2>/dev/null || echo "") + LAST_BRANCH=$(cat "$LAST_BRANCH_FILE" 2>/dev/null || echo "") + + if [ -n "$CURRENT_BRANCH" ] && [ -n "$LAST_BRANCH" ] && [ "$CURRENT_BRANCH" != "$LAST_BRANCH" ]; then + DATE=$(date +%Y-%m-%d) + FOLDER_NAME=$(echo "$LAST_BRANCH" | sed 's|^ralph/||') + ARCHIVE_FOLDER="$ARCHIVE_DIR/$DATE-$FOLDER_NAME" + + echo "Arhivare rulare anterioară: $LAST_BRANCH" + mkdir -p "$ARCHIVE_FOLDER" + [ -f "$PRD_FILE" ] && cp "$PRD_FILE" "$ARCHIVE_FOLDER/" + [ -f "$PROGRESS_FILE" ] && cp "$PROGRESS_FILE" "$ARCHIVE_FOLDER/" + echo " Arhivat în: $ARCHIVE_FOLDER" + + # Reset progress file + echo "# Ralph Progress Log" > "$PROGRESS_FILE" + echo "Started: $(date)" >> "$PROGRESS_FILE" + echo "Branch: $CURRENT_BRANCH" >> "$PROGRESS_FILE" + echo "---" >> "$PROGRESS_FILE" + fi +fi + +# Salvează branch-ul curent +if [ -f "$PRD_FILE" ]; then + CURRENT_BRANCH=$(jq -r '.branchName // empty' "$PRD_FILE" 2>/dev/null || echo "") + if [ -n "$CURRENT_BRANCH" ]; then + echo "$CURRENT_BRANCH" > "$LAST_BRANCH_FILE" + fi +fi + +# Creează directoare necesare +mkdir -p "$SCRIPT_DIR/logs" "$SCRIPT_DIR/archive" "$SCRIPT_DIR/screenshots" + +# Creează .gitignore dacă nu există +if [ ! -f "$PROJECT_DIR/.gitignore" ]; then + cat > "$PROJECT_DIR/.gitignore" << 'GITIGNORE' +# Python +__pycache__/ +*.py[cod] +*.pyo +*.egg-info/ +dist/ +build/ +.coverage +htmlcov/ +.pytest_cache/ + +# Virtual environment +venv/ +.venv/ + +# Ralph runtime +scripts/ralph/.ralph.pid +scripts/ralph/.last-branch +scripts/ralph/logs/ +scripts/ralph/screenshots/ +scripts/ralph/archive/ + +# IDE +.vscode/ +.idea/ + +# OS +.DS_Store +Thumbs.db +GITIGNORE + echo "Created .gitignore" +fi + +# Inițializare progress file dacă nu există +if [ ! -f "$PROGRESS_FILE" ]; then + echo "# Ralph Progress Log" > "$PROGRESS_FILE" + echo "Started: $(date)" >> "$PROGRESS_FILE" + echo "---" >> "$PROGRESS_FILE" +fi + +# Funcție pentru a verifica dacă toate story-urile sunt complete +check_all_complete() { + local incomplete=$(jq '[.userStories[] | select(.passes != true)] | length' "$PRD_FILE" 2>/dev/null || echo "999") + [ "$incomplete" -eq 0 ] +} + +# Afișare status inițial +echo "" +echo "=======================================================================" +echo " RALPH pentru Claude Code - Agent Autonom " +echo "=======================================================================" +PROJECT_NAME=$(jq -r '.projectName // "Unknown"' "$PRD_FILE") +BRANCH_NAME=$(jq -r '.branchName // "N/A"' "$PRD_FILE") +TOTAL_STORIES=$(jq '.userStories | length' "$PRD_FILE") +COMPLETE_STORIES=$(jq '[.userStories[] | select(.passes == true)] | length' "$PRD_FILE") +echo " Proiect: $PROJECT_NAME" +echo " Branch: $BRANCH_NAME" +echo " Stories: $COMPLETE_STORIES / $TOTAL_STORIES complete" +echo " Max iterații: $MAX_ITERATIONS" +echo " Screenshots: $SCREENSHOTS_DIR" +echo "=======================================================================" +echo "" + +# Verificare rapidă - poate toate sunt deja complete? +if check_all_complete; then + echo "Toate story-urile sunt deja complete!" + exit 0 +fi + +# Loop principal +for i in $(seq 1 $MAX_ITERATIONS); do + echo "" + echo "===================================================================" + echo " Ralph Iterația $i din $MAX_ITERATIONS" + echo "===================================================================" + + # Status curent + COMPLETE_NOW=$(jq '[.userStories[] | select(.passes == true)] | length' "$PRD_FILE") + NEXT_STORY=$(jq -r '[.userStories[] | select(.passes != true)] | sort_by(.priority) | .[0] | "\(.id): \(.title)"' "$PRD_FILE") + echo " Progress: $COMPLETE_NOW / $TOTAL_STORIES stories complete" + echo " Next: $NEXT_STORY" + echo "" + + # Pregătește prompt-ul cu context + FULL_PROMPT=$(cat <&1 | tee "$LOG_FILE" || true + OUTPUT=$(cat "$LOG_FILE") + + # Verifică dacă toate task-urile sunt complete + if echo "$OUTPUT" | grep -q "COMPLETE"; then + echo "" + echo "===================================================================" + echo " RALPH A TERMINAT TOATE TASK-URILE!" + echo " Completat la iterația $i din $MAX_ITERATIONS" + echo "===================================================================" + exit 0 + fi + + # Verifică și prin prd.json + if check_all_complete; then + echo "" + echo "===================================================================" + echo " TOATE STORY-URILE DIN PRD SUNT COMPLETE!" + echo "===================================================================" + exit 0 + fi + + echo " Iterația $i completă. Continuăm..." + sleep 2 +done + +echo "" +echo "===================================================================" +echo " Ralph a atins limita de iterații ($MAX_ITERATIONS)" +echo " Verifică progress.txt pentru status." +echo "===================================================================" +echo "" + +# Afișează stories incomplete +echo "Stories incomplete:" +jq -r '.userStories[] | select(.passes != true) | " - \(.id): \(.title)"' "$PRD_FILE" + +exit 1