Files
skill-seekers-reference/src/skill_seekers/cli/install_skill.py
2026-01-17 17:48:15 +00:00

179 lines
5.2 KiB
Python

#!/usr/bin/env python3
"""
Complete Skill Installation Workflow
One-command installation: fetch → scrape → enhance → package → upload
This CLI tool orchestrates the complete skill installation workflow by calling
the install_skill MCP tool.
Usage:
skill-seekers install --config react
skill-seekers install --config configs/custom.json --no-upload
skill-seekers install --config django --unlimited
skill-seekers install --config react --dry-run
Examples:
# Install React skill from official configs
skill-seekers install --config react
# Install from local config file
skill-seekers install --config configs/custom.json
# Install without uploading
skill-seekers install --config django --no-upload
# Preview workflow without executing
skill-seekers install --config react --dry-run
"""
import argparse
import asyncio
import sys
from pathlib import Path
# Add parent directory to path to import MCP server
sys.path.insert(0, str(Path(__file__).parent.parent))
# Import the MCP tool function (with lazy loading)
try:
from skill_seekers.mcp.server import install_skill_tool
MCP_AVAILABLE = True
except ImportError:
MCP_AVAILABLE = False
install_skill_tool = None
def main():
"""Main entry point for CLI"""
# Check MCP availability first
if not MCP_AVAILABLE:
print("\n❌ Error: MCP package not installed")
print("\nThe 'install' command requires MCP support.")
print("Install with:")
print(" pip install skill-seekers[mcp]")
print("\nOr use these alternatives:")
print(" skill-seekers scrape --config react")
print(" skill-seekers package output/react/")
print()
sys.exit(1)
parser = argparse.ArgumentParser(
description="Complete skill installation workflow (fetch → scrape → enhance → package → upload)",
formatter_class=argparse.RawDescriptionHelpFormatter,
epilog="""
Examples:
# Install React skill from official API
skill-seekers install --config react
# Install from local config file
skill-seekers install --config configs/custom.json
# Install without uploading
skill-seekers install --config django --no-upload
# Unlimited scraping (no page limits)
skill-seekers install --config godot --unlimited
# Preview workflow (dry run)
skill-seekers install --config react --dry-run
# Install for Gemini instead of Claude
skill-seekers install --config react --target gemini
# Install for OpenAI ChatGPT
skill-seekers install --config fastapi --target openai
Important:
- Enhancement is MANDATORY (30-60 sec) for quality (3/10→9/10)
- Total time: 20-45 minutes (mostly scraping)
- Multi-platform support: claude (default), gemini, openai, markdown
- Auto-uploads if API key is set (ANTHROPIC_API_KEY, GOOGLE_API_KEY, or OPENAI_API_KEY)
Phases:
1. Fetch config (if config name provided)
2. Scrape documentation
3. AI Enhancement (MANDATORY - no skip option)
4. Package for target platform (ZIP or tar.gz)
5. Upload to target platform (optional)
""",
)
parser.add_argument(
"--config",
required=True,
help="Config name (e.g., 'react') or path (e.g., 'configs/custom.json')",
)
parser.add_argument(
"--destination",
default="output",
help="Output directory for skill files (default: output/)",
)
parser.add_argument("--no-upload", action="store_true", help="Skip automatic upload to Claude")
parser.add_argument(
"--unlimited",
action="store_true",
help="Remove page limits during scraping (WARNING: Can take hours)",
)
parser.add_argument("--dry-run", action="store_true", help="Preview workflow without executing")
parser.add_argument(
"--target",
choices=["claude", "gemini", "openai", "markdown"],
default="claude",
help="Target LLM platform (default: claude)",
)
args = parser.parse_args()
# Determine if config is a name or path
config_arg = args.config
if config_arg.endswith(".json") or "/" in config_arg or "\\" in config_arg:
# It's a path
config_path = config_arg
config_name = None
else:
# It's a name
config_name = config_arg
config_path = None
# Build arguments for install_skill_tool
tool_args = {
"config_name": config_name,
"config_path": config_path,
"destination": args.destination,
"auto_upload": not args.no_upload,
"unlimited": args.unlimited,
"dry_run": args.dry_run,
"target": args.target,
}
# Run async tool
try:
result = asyncio.run(install_skill_tool(tool_args))
# Print output
for content in result:
print(content.text)
# Return success/failure based on output
output_text = result[0].text
if "" in output_text and "WORKFLOW COMPLETE" not in output_text:
return 1
return 0
except KeyboardInterrupt:
print("\n\n⚠️ Workflow interrupted by user")
return 130 # Standard exit code for SIGINT
except Exception as e:
print(f"\n\n❌ Unexpected error: {str(e)}")
return 1
if __name__ == "__main__":
sys.exit(main())