docs: 更新 HTTP 错误码、用户积分、占卜运行及用户资料协议文档
This commit is contained in:
Executable
+465
@@ -0,0 +1,465 @@
|
||||
#!/usr/bin/env python3
|
||||
"""
|
||||
Multi-Agent Pipeline: Start Worktree Agent.
|
||||
|
||||
Usage: python3 start.py <task-dir>
|
||||
Example: python3 start.py .trellis/tasks/01-21-my-task
|
||||
|
||||
This script:
|
||||
1. Creates worktree (if not exists) with dependency install
|
||||
2. Copies environment files (from worktree.yaml config)
|
||||
3. Sets .current-task in worktree
|
||||
4. Starts claude agent in background
|
||||
5. Registers agent to registry.json
|
||||
|
||||
Prerequisites:
|
||||
- task.json must exist with 'branch' field
|
||||
- agents/dispatch.md must exist (in .claude/, .cursor/, .iflow/, or .opencode/)
|
||||
|
||||
Configuration: .trellis/worktree.yaml
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
import json
|
||||
import os
|
||||
import shutil
|
||||
import subprocess
|
||||
import sys
|
||||
import uuid
|
||||
from pathlib import Path
|
||||
|
||||
# Add parent directory to path for imports
|
||||
sys.path.insert(0, str(Path(__file__).parent.parent))
|
||||
|
||||
from common.cli_adapter import CLIAdapter, get_cli_adapter
|
||||
from common.git_context import _run_git_command
|
||||
from common.paths import (
|
||||
DIR_WORKFLOW,
|
||||
FILE_CURRENT_TASK,
|
||||
FILE_TASK_JSON,
|
||||
get_repo_root,
|
||||
)
|
||||
from common.registry import (
|
||||
registry_add_agent,
|
||||
registry_get_file,
|
||||
)
|
||||
from common.worktree import (
|
||||
get_worktree_base_dir,
|
||||
get_worktree_config,
|
||||
get_worktree_copy_files,
|
||||
get_worktree_post_create_hooks,
|
||||
)
|
||||
|
||||
# =============================================================================
|
||||
# Colors
|
||||
# =============================================================================
|
||||
|
||||
|
||||
class Colors:
|
||||
RED = "\033[0;31m"
|
||||
GREEN = "\033[0;32m"
|
||||
YELLOW = "\033[1;33m"
|
||||
BLUE = "\033[0;34m"
|
||||
NC = "\033[0m"
|
||||
|
||||
|
||||
def log_info(msg: str) -> None:
|
||||
print(f"{Colors.BLUE}[INFO]{Colors.NC} {msg}")
|
||||
|
||||
|
||||
def log_success(msg: str) -> None:
|
||||
print(f"{Colors.GREEN}[SUCCESS]{Colors.NC} {msg}")
|
||||
|
||||
|
||||
def log_warn(msg: str) -> None:
|
||||
print(f"{Colors.YELLOW}[WARN]{Colors.NC} {msg}")
|
||||
|
||||
|
||||
def log_error(msg: str) -> None:
|
||||
print(f"{Colors.RED}[ERROR]{Colors.NC} {msg}")
|
||||
|
||||
|
||||
# =============================================================================
|
||||
# Helper Functions
|
||||
# =============================================================================
|
||||
|
||||
|
||||
def _read_json_file(path: Path) -> dict | None:
|
||||
"""Read and parse a JSON file."""
|
||||
try:
|
||||
return json.loads(path.read_text(encoding="utf-8"))
|
||||
except (FileNotFoundError, json.JSONDecodeError, OSError):
|
||||
return None
|
||||
|
||||
|
||||
def _write_json_file(path: Path, data: dict) -> bool:
|
||||
"""Write dict to JSON file."""
|
||||
try:
|
||||
path.write_text(
|
||||
json.dumps(data, indent=2, ensure_ascii=False), encoding="utf-8"
|
||||
)
|
||||
return True
|
||||
except (OSError, IOError):
|
||||
return False
|
||||
|
||||
|
||||
# =============================================================================
|
||||
# Constants
|
||||
# =============================================================================
|
||||
|
||||
DEFAULT_PLATFORM = "claude"
|
||||
|
||||
|
||||
# =============================================================================
|
||||
# Main
|
||||
# =============================================================================
|
||||
|
||||
|
||||
def main() -> int:
|
||||
"""Main entry point."""
|
||||
import argparse
|
||||
|
||||
parser = argparse.ArgumentParser(description="Multi-Agent Pipeline: Start Worktree Agent")
|
||||
parser.add_argument("task_dir", help="Task directory path")
|
||||
parser.add_argument(
|
||||
"--platform", "-p",
|
||||
choices=["claude", "cursor", "iflow", "opencode", "qoder"],
|
||||
default=DEFAULT_PLATFORM,
|
||||
help="Platform to use (default: claude)"
|
||||
)
|
||||
|
||||
args = parser.parse_args()
|
||||
task_dir_arg = args.task_dir
|
||||
platform = args.platform
|
||||
|
||||
# Initialize CLI adapter
|
||||
adapter = get_cli_adapter(platform)
|
||||
|
||||
project_root = get_repo_root()
|
||||
|
||||
# Normalize paths
|
||||
if task_dir_arg.startswith("/"):
|
||||
task_dir_relative = task_dir_arg[len(str(project_root)) + 1 :]
|
||||
task_dir_abs = Path(task_dir_arg)
|
||||
else:
|
||||
task_dir_relative = task_dir_arg
|
||||
task_dir_abs = project_root / task_dir_arg
|
||||
|
||||
task_json_path = task_dir_abs / FILE_TASK_JSON
|
||||
|
||||
# =============================================================================
|
||||
# Validation
|
||||
# =============================================================================
|
||||
if not task_json_path.is_file():
|
||||
log_error(f"task.json not found at {task_json_path}")
|
||||
return 1
|
||||
|
||||
dispatch_md = adapter.get_agent_path("dispatch", project_root)
|
||||
if not dispatch_md.is_file():
|
||||
log_error(f"dispatch.md not found at {dispatch_md}")
|
||||
log_info(f"Platform: {platform}")
|
||||
return 1
|
||||
|
||||
config_file = get_worktree_config(project_root)
|
||||
if not config_file.is_file():
|
||||
log_error(f"worktree.yaml not found at {config_file}")
|
||||
return 1
|
||||
|
||||
# =============================================================================
|
||||
# Read Task Config
|
||||
# =============================================================================
|
||||
print()
|
||||
print(f"{Colors.BLUE}=== Multi-Agent Pipeline: Start ==={Colors.NC}")
|
||||
log_info(f"Task: {task_dir_abs}")
|
||||
|
||||
task_data = _read_json_file(task_json_path)
|
||||
if not task_data:
|
||||
log_error("Failed to read task.json")
|
||||
return 1
|
||||
|
||||
branch = task_data.get("branch")
|
||||
task_name = task_data.get("name")
|
||||
task_status = task_data.get("status")
|
||||
worktree_path = task_data.get("worktree_path")
|
||||
|
||||
# Check if task was rejected
|
||||
if task_status == "rejected":
|
||||
log_error("Task was rejected by Plan Agent")
|
||||
rejected_file = task_dir_abs / "REJECTED.md"
|
||||
if rejected_file.is_file():
|
||||
print()
|
||||
print(f"{Colors.YELLOW}Rejection reason:{Colors.NC}")
|
||||
print(rejected_file.read_text(encoding="utf-8"))
|
||||
print()
|
||||
log_info(
|
||||
"To retry, delete this directory and run plan.py again with revised requirements"
|
||||
)
|
||||
return 1
|
||||
|
||||
# Check if prd.md exists (plan completed successfully)
|
||||
prd_file = task_dir_abs / "prd.md"
|
||||
if not prd_file.is_file():
|
||||
log_error("prd.md not found - Plan Agent may not have completed")
|
||||
log_info(f"Check plan log: {task_dir_abs}/.plan-log")
|
||||
return 1
|
||||
|
||||
if not branch:
|
||||
log_error("branch field not set in task.json")
|
||||
log_info("Please set branch field first, e.g.:")
|
||||
log_info(
|
||||
" jq '.branch = \"task/my-task\"' task.json > tmp && mv tmp task.json"
|
||||
)
|
||||
return 1
|
||||
|
||||
log_info(f"Branch: {branch}")
|
||||
log_info(f"Name: {task_name}")
|
||||
|
||||
# =============================================================================
|
||||
# Step 1: Create Worktree (if not exists)
|
||||
# =============================================================================
|
||||
if not worktree_path or not Path(worktree_path).is_dir():
|
||||
log_info("Step 1: Creating worktree...")
|
||||
|
||||
# Record current branch as base_branch (PR target)
|
||||
_, base_branch_out, _ = _run_git_command(
|
||||
["branch", "--show-current"], cwd=project_root
|
||||
)
|
||||
base_branch = base_branch_out.strip()
|
||||
log_info(f"Base branch (PR target): {base_branch}")
|
||||
|
||||
# Calculate worktree path
|
||||
worktree_base = get_worktree_base_dir(project_root)
|
||||
worktree_base.mkdir(parents=True, exist_ok=True)
|
||||
worktree_base = worktree_base.resolve()
|
||||
worktree_path_obj = worktree_base / branch
|
||||
worktree_path = str(worktree_path_obj)
|
||||
|
||||
# Create parent directory
|
||||
worktree_path_obj.parent.mkdir(parents=True, exist_ok=True)
|
||||
|
||||
# Create branch if not exists
|
||||
ret, _, _ = _run_git_command(
|
||||
["show-ref", "--verify", "--quiet", f"refs/heads/{branch}"],
|
||||
cwd=project_root,
|
||||
)
|
||||
if ret == 0:
|
||||
log_info("Branch exists, checking out...")
|
||||
ret, _, err = _run_git_command(
|
||||
["worktree", "add", worktree_path, branch], cwd=project_root
|
||||
)
|
||||
else:
|
||||
log_info(f"Creating new branch: {branch}")
|
||||
ret, _, err = _run_git_command(
|
||||
["worktree", "add", "-b", branch, worktree_path], cwd=project_root
|
||||
)
|
||||
|
||||
if ret != 0:
|
||||
log_error(f"Failed to create worktree: {err}")
|
||||
return 1
|
||||
|
||||
log_success(f"Worktree created: {worktree_path}")
|
||||
|
||||
# Update task.json with worktree_path and base_branch
|
||||
task_data["worktree_path"] = worktree_path
|
||||
task_data["base_branch"] = base_branch
|
||||
_write_json_file(task_json_path, task_data)
|
||||
|
||||
# ----- Copy environment files -----
|
||||
log_info("Copying environment files...")
|
||||
copy_list = get_worktree_copy_files(project_root)
|
||||
copy_count = 0
|
||||
|
||||
for item in copy_list:
|
||||
if not item:
|
||||
continue
|
||||
|
||||
source = project_root / item
|
||||
target = Path(worktree_path) / item
|
||||
|
||||
if source.is_file():
|
||||
target.parent.mkdir(parents=True, exist_ok=True)
|
||||
shutil.copy2(str(source), str(target))
|
||||
copy_count += 1
|
||||
|
||||
if copy_count > 0:
|
||||
log_success(f"Copied {copy_count} file(s)")
|
||||
|
||||
# ----- Copy task directory (may not be committed yet) -----
|
||||
log_info("Copying task directory...")
|
||||
task_target_dir = Path(worktree_path) / task_dir_relative
|
||||
task_target_dir.parent.mkdir(parents=True, exist_ok=True)
|
||||
if task_target_dir.exists():
|
||||
shutil.rmtree(str(task_target_dir))
|
||||
shutil.copytree(str(task_dir_abs), str(task_target_dir))
|
||||
log_success("Task directory copied to worktree")
|
||||
|
||||
# ----- Run post_create hooks -----
|
||||
log_info("Running post_create hooks...")
|
||||
post_create = get_worktree_post_create_hooks(project_root)
|
||||
hook_count = 0
|
||||
|
||||
for cmd in post_create:
|
||||
if not cmd:
|
||||
continue
|
||||
|
||||
log_info(f" Running: {cmd}")
|
||||
ret = subprocess.run(cmd, shell=True, cwd=worktree_path)
|
||||
if ret.returncode != 0:
|
||||
log_error(f"Hook failed: {cmd}")
|
||||
return 1
|
||||
hook_count += 1
|
||||
|
||||
if hook_count > 0:
|
||||
log_success(f"Ran {hook_count} hook(s)")
|
||||
else:
|
||||
log_info(f"Step 1: Using existing worktree: {worktree_path}")
|
||||
|
||||
# =============================================================================
|
||||
# Step 2: Set .current-task in Worktree
|
||||
# =============================================================================
|
||||
log_info("Step 2: Setting current task in worktree...")
|
||||
|
||||
worktree_workflow_dir = Path(worktree_path) / DIR_WORKFLOW
|
||||
worktree_workflow_dir.mkdir(parents=True, exist_ok=True)
|
||||
|
||||
current_task_file = worktree_workflow_dir / FILE_CURRENT_TASK
|
||||
current_task_file.write_text(task_dir_relative, encoding="utf-8")
|
||||
log_success(f"Current task set: {task_dir_relative}")
|
||||
|
||||
# =============================================================================
|
||||
# Step 3: Prepare and Start Claude Agent
|
||||
# =============================================================================
|
||||
log_info(f"Step 3: Starting {adapter.cli_name} agent...")
|
||||
|
||||
# Update task status
|
||||
task_data["status"] = "in_progress"
|
||||
_write_json_file(task_json_path, task_data)
|
||||
|
||||
log_file = Path(worktree_path) / ".agent-log"
|
||||
session_id_file = Path(worktree_path) / ".session-id"
|
||||
|
||||
log_file.touch()
|
||||
|
||||
# Generate session ID for resume support (Claude Code only)
|
||||
# OpenCode generates its own session ID, we'll extract it from logs later
|
||||
if adapter.supports_session_id_on_create:
|
||||
session_id = str(uuid.uuid4()).lower()
|
||||
session_id_file.write_text(session_id, encoding="utf-8")
|
||||
log_info(f"Session ID: {session_id}")
|
||||
else:
|
||||
session_id = None # Will be extracted from logs after startup
|
||||
log_info("Session ID will be extracted from logs after startup")
|
||||
|
||||
# Get proxy environment variables
|
||||
https_proxy = os.environ.get("https_proxy", "")
|
||||
http_proxy = os.environ.get("http_proxy", "")
|
||||
all_proxy = os.environ.get("all_proxy", "")
|
||||
|
||||
# Start agent in background (cross-platform, no shell script needed)
|
||||
env = os.environ.copy()
|
||||
env["https_proxy"] = https_proxy
|
||||
env["http_proxy"] = http_proxy
|
||||
env["all_proxy"] = all_proxy
|
||||
|
||||
# Clear nested-session detection so the new CLI process can start
|
||||
# (when this script runs inside a Claude Code session, CLAUDECODE=1 is inherited)
|
||||
env.pop("CLAUDECODE", None)
|
||||
|
||||
# Set non-interactive env var based on platform
|
||||
env.update(adapter.get_non_interactive_env())
|
||||
|
||||
# Build CLI command using adapter
|
||||
# Note: Use explicit prompt to avoid confusion with CI/CD pipelines
|
||||
# Also remind the model to follow its agent definition for better cross-model compatibility
|
||||
cli_cmd = adapter.build_run_command(
|
||||
agent="dispatch",
|
||||
prompt="Follow your agent instructions to execute the task workflow. Start by reading .trellis/.current-task to get the task directory, then execute each action in task.json next_action array in order.",
|
||||
session_id=session_id if adapter.supports_session_id_on_create else None,
|
||||
skip_permissions=True,
|
||||
verbose=True,
|
||||
json_output=True,
|
||||
)
|
||||
|
||||
with log_file.open("w") as log_f:
|
||||
# Use shell=False for cross-platform compatibility
|
||||
# creationflags for Windows, start_new_session for Unix
|
||||
popen_kwargs = {
|
||||
"stdout": log_f,
|
||||
"stderr": subprocess.STDOUT,
|
||||
"cwd": worktree_path,
|
||||
"env": env,
|
||||
}
|
||||
if sys.platform == "win32":
|
||||
popen_kwargs["creationflags"] = subprocess.CREATE_NEW_PROCESS_GROUP
|
||||
else:
|
||||
popen_kwargs["start_new_session"] = True
|
||||
|
||||
process = subprocess.Popen(cli_cmd, **popen_kwargs)
|
||||
agent_pid = process.pid
|
||||
|
||||
log_success(f"Agent started with PID: {agent_pid}")
|
||||
|
||||
# For OpenCode: extract session ID from logs after startup
|
||||
if not adapter.supports_session_id_on_create:
|
||||
import time
|
||||
log_info("Waiting for session ID from logs...")
|
||||
# Wait a bit for the log to have session ID
|
||||
for _ in range(10): # Try for up to 5 seconds
|
||||
time.sleep(0.5)
|
||||
try:
|
||||
log_content = log_file.read_text(encoding="utf-8", errors="replace")
|
||||
session_id = adapter.extract_session_id_from_log(log_content)
|
||||
if session_id:
|
||||
session_id_file.write_text(session_id, encoding="utf-8")
|
||||
log_success(f"Session ID extracted: {session_id}")
|
||||
break
|
||||
except Exception:
|
||||
pass
|
||||
else:
|
||||
log_warn("Could not extract session ID from logs")
|
||||
session_id = "unknown"
|
||||
|
||||
# =============================================================================
|
||||
# Step 4: Register to Registry (in main repo, not worktree)
|
||||
# =============================================================================
|
||||
log_info("Step 4: Registering agent to registry...")
|
||||
|
||||
# Generate agent ID
|
||||
task_id = task_data.get("id")
|
||||
if not task_id:
|
||||
task_id = branch.replace("/", "-")
|
||||
|
||||
registry_add_agent(
|
||||
task_id, worktree_path, agent_pid, task_dir_relative, project_root, platform
|
||||
)
|
||||
|
||||
log_success(f"Agent registered: {task_id}")
|
||||
|
||||
# =============================================================================
|
||||
# Summary
|
||||
# =============================================================================
|
||||
print()
|
||||
print(f"{Colors.GREEN}=== Agent Started ==={Colors.NC}")
|
||||
print()
|
||||
print(f" ID: {task_id}")
|
||||
print(f" PID: {agent_pid}")
|
||||
print(f" Session: {session_id}")
|
||||
print(f" Worktree: {worktree_path}")
|
||||
print(f" Task: {task_dir_relative}")
|
||||
print(f" Log: {log_file}")
|
||||
print(f" Registry: {registry_get_file(project_root)}")
|
||||
print()
|
||||
print(f"{Colors.YELLOW}To monitor:{Colors.NC} tail -f {log_file}")
|
||||
print(f"{Colors.YELLOW}To stop:{Colors.NC} kill {agent_pid}")
|
||||
if session_id and session_id != "unknown":
|
||||
resume_cmd = adapter.get_resume_command_str(session_id, cwd=worktree_path)
|
||||
print(f"{Colors.YELLOW}To resume:{Colors.NC} {resume_cmd}")
|
||||
else:
|
||||
print(f"{Colors.YELLOW}To resume:{Colors.NC} (session ID not available)")
|
||||
|
||||
return 0
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
sys.exit(main())
|
||||
Reference in New Issue
Block a user