mirror of
https://github.com/microsoft/SkillOpt.git
synced 2026-07-03 14:02:58 +08:00
Bug fixes: - #52: bundle run-sleep.sh in Claude Code plugin + 4-level fallback - #58: add skillopt-sleep console script entry point in pyproject.toml - #62: filter headless claude -p replay sessions from harvest Plugin sync (Claude Code / Codex / Copilot / OpenClaw): - Document all 22 CLI flags, 7 actions, 4 backends across all SKILL.md files - Document config keys (preferences, gate_mode, dream_rollouts, etc.) - Document memory consolidation (evolve_memory / evolve_skill) - Add schedule/unschedule to all plugins - Copilot MCP: expand schema from 3 → 16 params + schedule tools - OpenClaw: add schedule/unschedule subcommands via shared scheduler Tests: - Cross-plugin parity test (prevents future feature drift) - MCP schema completeness test Co-Authored-By: Claude Fable 5 <noreply@anthropic.com>
181 lines
7.3 KiB
Python
Executable File
181 lines
7.3 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
"""SkillOpt-Sleep — minimal MCP server (stdio, stdlib-only).
|
|
|
|
Exposes the sleep engine as MCP tools so any MCP-capable client (GitHub Copilot
|
|
CLI / VS Code, Claude Desktop, etc.) can drive it. No third-party deps: speaks
|
|
JSON-RPC 2.0 over stdio with just the handful of MCP methods clients need.
|
|
|
|
Tools exposed:
|
|
- sleep_status : how many nights have run + the latest staged proposal
|
|
- sleep_dry_run : harvest+mine+replay, report only (no staging)
|
|
- sleep_run : full cycle, stages a proposal (nothing live changes)
|
|
- sleep_adopt : apply the latest staged proposal (with backup)
|
|
- sleep_harvest : debug — list mined recurring tasks
|
|
|
|
Each tool shells out to `python -m skillopt_sleep <action> ...` and returns its
|
|
stdout. Configure your client to launch: python plugins/copilot/mcp_server.py
|
|
"""
|
|
from __future__ import annotations
|
|
|
|
import json
|
|
import os
|
|
import subprocess
|
|
import sys
|
|
|
|
REPO_ROOT = os.environ.get("SKILLOPT_SLEEP_REPO") or os.path.abspath(
|
|
os.path.join(os.path.dirname(__file__), "..", "..")
|
|
)
|
|
PROTOCOL_VERSION = "2024-11-05"
|
|
|
|
TOOLS = [
|
|
{"name": "sleep_status", "action": "status",
|
|
"description": "Show how many SkillOpt-Sleep nights have run and the latest staged proposal."},
|
|
{"name": "sleep_dry_run", "action": "dry-run",
|
|
"description": "Preview a sleep cycle (harvest+mine+replay) without staging anything."},
|
|
{"name": "sleep_run", "action": "run",
|
|
"description": "Run a full sleep cycle; stages a reviewed proposal. Nothing live changes until adopt."},
|
|
{"name": "sleep_adopt", "action": "adopt",
|
|
"description": "Apply the latest staged proposal to CLAUDE.md/SKILL.md (backs up first)."},
|
|
{"name": "sleep_harvest", "action": "harvest",
|
|
"description": "Debug: list the recurring tasks mined from recent sessions."},
|
|
{"name": "sleep_schedule", "action": "schedule",
|
|
"description": "Install a nightly cron entry to run the sleep cycle automatically."},
|
|
{"name": "sleep_unschedule", "action": "unschedule",
|
|
"description": "Remove the nightly cron entry for a project."},
|
|
]
|
|
_BY_NAME = {t["name"]: t for t in TOOLS}
|
|
|
|
_TOOL_SCHEMA = {
|
|
"type": "object",
|
|
"properties": {
|
|
"project": {"type": "string",
|
|
"description": "Project dir to evolve (default: cwd)."},
|
|
"backend": {"type": "string", "enum": ["mock", "claude", "codex", "copilot"],
|
|
"description": "mock = no API spend (default); claude/codex/copilot = real."},
|
|
"scope": {"type": "string", "enum": ["invoked", "all"],
|
|
"description": "Harvest scope (default: invoked project only)."},
|
|
"source": {"type": "string", "enum": ["claude", "codex", "auto"],
|
|
"description": "Transcript source (default: claude)."},
|
|
"model": {"type": "string",
|
|
"description": "Backend-specific model override."},
|
|
"tasks_file": {"type": "string",
|
|
"description": "Path to reviewed TaskRecord JSON (skips harvest)."},
|
|
"target_skill_path": {"type": "string",
|
|
"description": "Explicit SKILL.md path to evolve/stage/adopt."},
|
|
"progress": {"type": "boolean",
|
|
"description": "Print phase progress to stderr."},
|
|
"max_sessions": {"type": "integer",
|
|
"description": "Cap harvested sessions per run."},
|
|
"max_tasks": {"type": "integer",
|
|
"description": "Cap mined tasks per run."},
|
|
"lookback_hours": {"type": "integer",
|
|
"description": "Harvest window in hours (default: 72)."},
|
|
"auto_adopt": {"type": "boolean",
|
|
"description": "Auto-adopt if gate passes (default: false)."},
|
|
"json": {"type": "boolean",
|
|
"description": "Return machine-readable JSON output."},
|
|
"edit_budget": {"type": "integer",
|
|
"description": "Max bounded edits per night (default: 4)."},
|
|
"hour": {"type": "integer",
|
|
"description": "Hour for schedule (0-23, default: 3)."},
|
|
"minute": {"type": "integer",
|
|
"description": "Minute for schedule (0-59, default: 17)."},
|
|
},
|
|
"additionalProperties": False,
|
|
}
|
|
|
|
|
|
def _run_engine(action: str, args: dict) -> str:
|
|
py = sys.executable or "python3"
|
|
cmd = [py, "-m", "skillopt_sleep", action]
|
|
# String-valued flags
|
|
for flag, key in [
|
|
("--project", "project"), ("--backend", "backend"),
|
|
("--scope", "scope"), ("--source", "source"),
|
|
("--model", "model"), ("--tasks-file", "tasks_file"),
|
|
("--target-skill-path", "target_skill_path"),
|
|
]:
|
|
val = args.get(key)
|
|
if val:
|
|
cmd += [flag, str(val)]
|
|
# Integer-valued flags
|
|
for flag, key in [
|
|
("--max-sessions", "max_sessions"), ("--max-tasks", "max_tasks"),
|
|
("--lookback-hours", "lookback_hours"), ("--edit-budget", "edit_budget"),
|
|
("--hour", "hour"), ("--minute", "minute"),
|
|
]:
|
|
val = args.get(key)
|
|
if val is not None:
|
|
cmd += [flag, str(int(val))]
|
|
# Boolean flags
|
|
for flag, key in [
|
|
("--progress", "progress"), ("--auto-adopt", "auto_adopt"),
|
|
("--json", "json"),
|
|
]:
|
|
if args.get(key):
|
|
cmd.append(flag)
|
|
try:
|
|
proc = subprocess.run(cmd, cwd=REPO_ROOT, capture_output=True, text=True, timeout=3600)
|
|
except Exception as e:
|
|
return f"[error] failed to run engine: {e}"
|
|
out = (proc.stdout or "").strip()
|
|
err = (proc.stderr or "").strip()
|
|
return out + (("\n[stderr]\n" + err) if err else "")
|
|
|
|
|
|
def _result(id_, result):
|
|
return {"jsonrpc": "2.0", "id": id_, "result": result}
|
|
|
|
|
|
def _error(id_, code, message):
|
|
return {"jsonrpc": "2.0", "id": id_, "error": {"code": code, "message": message}}
|
|
|
|
|
|
def handle(req: dict):
|
|
method = req.get("method")
|
|
id_ = req.get("id")
|
|
if method == "initialize":
|
|
return _result(id_, {
|
|
"protocolVersion": PROTOCOL_VERSION,
|
|
"capabilities": {"tools": {}},
|
|
"serverInfo": {"name": "skillopt-sleep", "version": "0.1.0"},
|
|
})
|
|
if method in ("notifications/initialized", "initialized"):
|
|
return None # notification, no response
|
|
if method == "tools/list":
|
|
return _result(id_, {"tools": [
|
|
{"name": t["name"], "description": t["description"], "inputSchema": _TOOL_SCHEMA}
|
|
for t in TOOLS
|
|
]})
|
|
if method == "tools/call":
|
|
params = req.get("params") or {}
|
|
name = params.get("name")
|
|
tool = _BY_NAME.get(name)
|
|
if not tool:
|
|
return _error(id_, -32602, f"unknown tool: {name}")
|
|
text = _run_engine(tool["action"], params.get("arguments") or {})
|
|
return _result(id_, {"content": [{"type": "text", "text": text}]})
|
|
if method == "ping":
|
|
return _result(id_, {})
|
|
return _error(id_, -32601, f"method not found: {method}")
|
|
|
|
|
|
def main() -> int:
|
|
for line in sys.stdin:
|
|
line = line.strip()
|
|
if not line:
|
|
continue
|
|
try:
|
|
req = json.loads(line)
|
|
except Exception:
|
|
continue
|
|
resp = handle(req)
|
|
if resp is not None:
|
|
sys.stdout.write(json.dumps(resp) + "\n")
|
|
sys.stdout.flush()
|
|
return 0
|
|
|
|
|
|
if __name__ == "__main__":
|
|
raise SystemExit(main())
|