mirror of
https://fastgit.cc/github.com/HKUDS/CLI-Anything
synced 2026-04-30 22:02:01 +08:00
7.3 KiB
7.3 KiB
cli-anything:list Command
List all available CLI-Anything tools (installed and generated).
Usage
/cli-anything:list [--path <directory>] [--depth <n>] [--json]
Options
--path <directory>- Directory to search for generated CLIs (default: current directory)--depth <n>- Maximum recursion depth for scanning (default: unlimited). Use0for current directory only,1for one level deep, etc.--json- Output in JSON format for machine parsing
What This Command Does
Displays all CLI-Anything tools available in the system:
1. Installed CLIs
Uses importlib.metadata to find installed cli-anything-* packages:
- Pattern: package name starts with
cli-anything- - Extracts: software name, version, entry point
from importlib.metadata import distributions
installed = {}
for dist in distributions():
name = dist.metadata.get("Name", "")
if name.startswith("cli-anything-"):
software = name.replace("cli-anything-", "")
version = dist.version
# Find executable via entry points or shutil.which
executable = shutil.which(f"cli-anything-{software}")
installed[software] = {
"status": "installed",
"version": version,
"executable": executable
}
2. Generated CLIs
Uses glob to find local CLI directories:
- Pattern:
**/agent-harness/cli_anything/*/__init__.py(or depth-limited variant) - Extracts: software name, version (from setup.py), source path
- Status:
generated
from pathlib import Path
import glob
import re
search_path = args.get("path", ".")
max_depth = args.get("depth", None) # None means unlimited
generated = {}
def extract_version_from_setup(setup_path):
"""Extract version from setup.py using regex."""
try:
content = Path(setup_path).read_text()
match = re.search(r'version\s*=\s*["\']([^"\']+)["\']', content)
return match.group(1) if match else None
except:
return None
def build_glob_patterns(base_path, depth):
"""Build list of glob patterns for depths 0 through max_depth.
Returns multiple patterns so that --depth 2 finds tools at depth 0, 1, AND 2.
"""
base = Path(base_path)
suffix = "agent-harness/cli_anything/*/__init__.py"
if depth is None:
# Unlimited depth: use **
return [str(base / "**" / suffix)]
# Generate patterns for all depths from 0 to max_depth
patterns = []
for d in range(depth + 1):
if d == 0:
# depth 0: look in current directory
patterns.append(str(base / suffix))
else:
# depth N: look N levels deep
prefix = "/".join(["*"] * d)
patterns.append(str(base / prefix / suffix))
return patterns
patterns = build_glob_patterns(search_path, max_depth)
for pattern in patterns:
for init_file in glob.glob(pattern, recursive=True):
parts = Path(init_file).parts
# Find cli_anything/<software> pattern
for i, p in enumerate(parts):
if p == "cli_anything" and i + 1 < len(parts):
software = parts[i + 1]
# Get agent-harness directory as source
agent_harness_idx = parts.index("agent-harness") if "agent-harness" in parts else i - 1
source = str(Path(*parts[:agent_harness_idx + 2])) # up to agent-harness
# Extract version from setup.py (setup.py is in agent-harness/, not cli_anything/)
setup_path = Path(*parts[:agent_harness_idx + 1]) / "setup.py"
version = extract_version_from_setup(setup_path)
generated[software] = {
"status": "generated",
"version": version,
"executable": None,
"source": source
}
break
3. Merge Results
- Deduplicate by software name
- If both installed and generated: show
installedstatus with both paths - The
sourcefield shows where the generated code is (even for installed)
Output Formats
Table Format (default)
CLI-Anything Tools (found 5)
Name Status Version Source
──────────────────────────────────────────────────────────────
gimp installed 1.0.0 ./gimp/agent-harness
blender installed 1.0.0 ./blender/agent-harness
inkscape generated 1.0.0 ./inkscape/agent-harness
audacity generated 1.0.0 ./audacity/agent-harness
libreoffice generated 1.0.0 ./libreoffice/agent-harness
JSON Format (--json)
{
"tools": [
{
"name": "gimp",
"status": "installed",
"version": "1.0.0",
"executable": "/usr/local/bin/cli-anything-gimp",
"source": "./gimp/agent-harness"
},
{
"name": "inkscape",
"status": "generated",
"version": "1.0.0",
"executable": null,
"source": "./inkscape/agent-harness"
}
],
"total": 2,
"installed": 1,
"generated_only": 1
}
Error Handling
| Scenario | Action |
|---|---|
| No CLIs found | Show "No CLI-Anything tools found" message |
| Invalid --path | Show error: "Path not found: " |
| Permission denied | Skip directory, continue scanning, show warning |
Implementation Steps
When this command is invoked, the agent should:
-
Parse arguments
- Extract
--pathvalue (default:.) - Extract
--depthvalue (default:Nonefor unlimited recursion) - Extract
--jsonflag (default: false)
- Extract
-
Validate path exists
- If
--pathspecified and doesn't exist, show error and exit
- If
-
Scan installed CLIs
- Use
importlib.metadata.distributions()to find all packages - Filter for packages starting with
cli-anything- - Extract name, version, find executable path
- Use
-
Scan generated CLIs
- Build glob pattern based on depth parameter
- Use
glob.glob(pattern, recursive=True) - Parse directory structure to extract software name
- Calculate relative path from current directory
-
Merge results
- Create dict keyed by software name
- Prefer installed data when both exist
- Keep source path from generated if available
-
Format output
- If
--json: output JSON to stdout - Otherwise: format as table with proper alignment
- If
-
Print results
- Show summary line with count
- Show table or JSON
Examples
# List all tools in current directory (unlimited depth)
/cli-anything:list
# List tools with depth limit (only scan 2 levels deep)
/cli-anything:list --depth 2
# List tools in current directory only (no recursion)
/cli-anything:list --depth 0
# List tools with JSON output
/cli-anything:list --json
# Search a specific directory with depth limit
/cli-anything:list --path /projects/my-tools --depth 3
# Combined
/cli-anything:list --path ./output --depth 2 --json
Notes
--depthcontrols how many directory levels to descend from the search path- Default depth is unlimited (
**glob pattern) - CLI-Anything tools typically need at least 3-4 levels to find
agent-harness/cli_anything/software/__init__.py - Relative paths are preferred for readability
- The command should work without any external dependencies beyond Python stdlib