kanban API: comment endpoint + usage docs, notification includes curl example
This commit is contained in:
@@ -0,0 +1,231 @@
|
||||
#!/usr/bin/env python3
|
||||
"""Kanban HTTP API — 轻量 REST 接口,供跨机器 Agent 读写 kanban。
|
||||
不走 Hermes CLI,直接操作 SQLite,零依赖。"""
|
||||
import http.server
|
||||
import json
|
||||
import sqlite3
|
||||
import urllib.parse
|
||||
import os
|
||||
from datetime import datetime
|
||||
|
||||
KANBAN_DB = os.path.expanduser("~/.hermes/kanban.db")
|
||||
HOST = "0.0.0.0"
|
||||
PORT = 9580
|
||||
|
||||
|
||||
def db():
|
||||
return sqlite3.connect(KANBAN_DB)
|
||||
|
||||
|
||||
class KanbanAPI(http.server.BaseHTTPRequestHandler):
|
||||
def _json(self, data, status=200):
|
||||
self.send_response(status)
|
||||
self.send_header("Content-Type", "application/json; charset=utf-8")
|
||||
self.send_header("Access-Control-Allow-Origin", "*")
|
||||
self.end_headers()
|
||||
self.wfile.write(json.dumps(data, ensure_ascii=False).encode())
|
||||
|
||||
def _read_body(self):
|
||||
length = int(self.headers.get("Content-Length", 0))
|
||||
return json.loads(self.rfile.read(length)) if length else {}
|
||||
|
||||
def do_POST(self):
|
||||
path = urllib.parse.urlparse(self.path).path.rstrip("/")
|
||||
body = self._read_body()
|
||||
|
||||
if path == "/kanban/create":
|
||||
title = body.get("title", "")
|
||||
assignee = body.get("assignee", "")
|
||||
body_text = body.get("body", "")
|
||||
if not title:
|
||||
return self._json({"error": "title required"}, 400)
|
||||
|
||||
# Use hermes CLI to create the card
|
||||
import subprocess
|
||||
cmd = ["hermes", "kanban", "create", title]
|
||||
if assignee:
|
||||
cmd += ["--assignee", assignee]
|
||||
if body_text:
|
||||
cmd += ["--body", body_text]
|
||||
r = subprocess.run(cmd, capture_output=True, text=True, timeout=30)
|
||||
|
||||
if r.returncode != 0:
|
||||
return self._json({"error": r.stderr.strip()}, 500)
|
||||
|
||||
# Parse output: "Created t_xxxxx (ready, assignee=xxx)"
|
||||
task_id = None
|
||||
for word in r.stdout.split():
|
||||
if word.startswith("t_"):
|
||||
task_id = word.rstrip(",")
|
||||
break
|
||||
|
||||
return self._json({"ok": True, "task_id": task_id, "raw": r.stdout.strip()})
|
||||
|
||||
elif path == "/kanban/create_xmpp":
|
||||
# 创建卡片并 XMPP 通知核心群
|
||||
title = body.get("title", "")
|
||||
assignee = body.get("assignee", "")
|
||||
body_text = body.get("body", "")
|
||||
if not title:
|
||||
return self._json({"error": "title required"}, 400)
|
||||
|
||||
import subprocess
|
||||
cmd = ["hermes", "kanban", "create", title]
|
||||
if assignee:
|
||||
cmd += ["--assignee", assignee]
|
||||
if body_text:
|
||||
cmd += ["--body", body_text]
|
||||
r = subprocess.run(cmd, capture_output=True, text=True, timeout=30)
|
||||
if r.returncode != 0:
|
||||
return self._json({"error": r.stderr.strip()}, 500)
|
||||
|
||||
task_id = None
|
||||
for word in r.stdout.split():
|
||||
if word.startswith("t_"):
|
||||
task_id = word.rstrip(",")
|
||||
break
|
||||
|
||||
# 发 XMPP 通知到核心群(走 mohe bot 的 5804 桥)
|
||||
xmpp_msg = f"[Kanban] 新任务 {task_id}: {title}"
|
||||
if assignee:
|
||||
xmpp_msg += f" → 指派给 {assignee}"
|
||||
if body_text:
|
||||
xmpp_msg += f"\n{body_text[:200]}"
|
||||
xmpp_msg += f"\n📌 评论: curl -X POST http://192.168.1.246:9580/kanban/comment -d '{{\"id\":\"{task_id}\",\"author\":\"你的名字\",\"body\":\"...\"}}'"
|
||||
|
||||
try:
|
||||
import urllib.request as _ur
|
||||
_noti = json.dumps({
|
||||
"to": "coregroup@conference.yoin.fun",
|
||||
"body": xmpp_msg
|
||||
}).encode()
|
||||
_req = _ur.Request("http://127.0.0.1:5804", data=_noti, method="POST")
|
||||
_req.add_header("Content-Type", "application/json")
|
||||
_ur.urlopen(_req, timeout=5)
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
return self._json({"ok": True, "task_id": task_id, "notified": "coregroup"})
|
||||
|
||||
elif path == "/kanban/list":
|
||||
status_filter = body.get("status")
|
||||
assignee_filter = body.get("assignee")
|
||||
|
||||
d = db()
|
||||
query = "SELECT id, title, body, status, assignee, created_by, created_at FROM tasks"
|
||||
params = []
|
||||
conditions = []
|
||||
if status_filter:
|
||||
conditions.append("status = ?")
|
||||
params.append(status_filter)
|
||||
if assignee_filter:
|
||||
conditions.append("assignee = ?")
|
||||
params.append(assignee_filter)
|
||||
if conditions:
|
||||
query += " WHERE " + " AND ".join(conditions)
|
||||
query += " ORDER BY created_at DESC"
|
||||
|
||||
rows = d.execute(query, params).fetchall()
|
||||
d.close()
|
||||
|
||||
tasks = []
|
||||
for r in rows:
|
||||
tasks.append({
|
||||
"id": r[0], "title": r[1], "body": r[2], "status": r[3],
|
||||
"assignee": r[4], "created_by": r[5], "created_at": r[6]
|
||||
})
|
||||
return self._json({"tasks": tasks})
|
||||
|
||||
elif path == "/kanban/comment":
|
||||
task_id = body.get("id")
|
||||
author = body.get("author", "unknown")
|
||||
text = body.get("body", "")
|
||||
if not task_id or not text:
|
||||
return self._json({"error": "id and body required"}, 400)
|
||||
d = db()
|
||||
d.execute(
|
||||
"INSERT INTO task_comments (task_id, author, body, created_at) VALUES (?, ?, ?, ?)",
|
||||
(task_id, author, text, int(__import__("time").time()))
|
||||
)
|
||||
d.commit()
|
||||
d.close()
|
||||
# 通知核心群
|
||||
try:
|
||||
_c_msg = f"[Kanban] {author} 评论了 {task_id}: {text[:100]}"
|
||||
_c_noti = json.dumps({"to": "coregroup@conference.yoin.fun", "body": _c_msg}).encode()
|
||||
_c_req = __import__("urllib.request").Request("http://127.0.0.1:5804", data=_c_noti, method="POST")
|
||||
_c_req.add_header("Content-Type", "application/json")
|
||||
__import__("urllib.request").urlopen(_c_req, timeout=5)
|
||||
except Exception:
|
||||
pass
|
||||
return self._json({"ok": True})
|
||||
|
||||
elif path == "/kanban/comments":
|
||||
task_id = body.get("id")
|
||||
if not task_id:
|
||||
return self._json({"error": "id required"}, 400)
|
||||
d = db()
|
||||
rows = d.execute(
|
||||
"SELECT id, author, body, created_at FROM task_comments WHERE task_id = ? ORDER BY created_at",
|
||||
(task_id,)
|
||||
).fetchall()
|
||||
d.close()
|
||||
comments = [{"id": r[0], "author": r[1], "body": r[2], "created_at": r[3]} for r in rows]
|
||||
return self._json({"comments": comments})
|
||||
|
||||
elif path == "/kanban/update":
|
||||
task_id = body.get("id")
|
||||
if not task_id:
|
||||
return self._json({"error": "id required"}, 400)
|
||||
|
||||
updates = {}
|
||||
for key in ("status", "assignee"):
|
||||
if key in body:
|
||||
updates[key] = body[key]
|
||||
if not updates:
|
||||
return self._json({"error": "nothing to update"}, 400)
|
||||
|
||||
d = db()
|
||||
set_clause = ", ".join(f"{k} = ?" for k in updates)
|
||||
d.execute(f"UPDATE tasks SET {set_clause} WHERE id = ?",
|
||||
list(updates.values()) + [task_id])
|
||||
d.commit()
|
||||
d.close()
|
||||
return self._json({"ok": True})
|
||||
|
||||
elif path == "/kanban/show":
|
||||
task_id = body.get("id")
|
||||
if not task_id:
|
||||
return self._json({"error": "id required"}, 400)
|
||||
d = db()
|
||||
row = d.execute(
|
||||
"SELECT id, title, body, status, assignee, created_by, created_at FROM tasks WHERE id = ?",
|
||||
(task_id,)
|
||||
).fetchone()
|
||||
d.close()
|
||||
if not row:
|
||||
return self._json({"error": "not found"}, 404)
|
||||
return self._json({
|
||||
"id": row[0], "title": row[1], "body": row[2],
|
||||
"status": row[3], "assignee": row[4],
|
||||
"created_by": row[5], "created_at": row[6]
|
||||
})
|
||||
|
||||
else:
|
||||
return self._json({"error": "not found"}, 404)
|
||||
|
||||
def do_OPTIONS(self):
|
||||
self.send_response(204)
|
||||
self.send_header("Access-Control-Allow-Origin", "*")
|
||||
self.send_header("Access-Control-Allow-Methods", "POST, OPTIONS")
|
||||
self.send_header("Access-Control-Allow-Headers", "Content-Type")
|
||||
self.end_headers()
|
||||
|
||||
def log_message(self, fmt, *args):
|
||||
print(f"[{datetime.now().strftime('%H:%M:%S')}] {args[0]}", flush=True)
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
server = http.server.HTTPServer((HOST, PORT), KanbanAPI)
|
||||
print(f"Kanban API running on http://{HOST}:{PORT}")
|
||||
server.serve_forever()
|
||||
Reference in New Issue
Block a user