luminos/luminos.py

163 lines
5.8 KiB
Python
Raw Normal View History

#!/usr/bin/env python3
"""Luminos — file system intelligence tool."""
import argparse
import json
import os
import shutil
import sys
from luminos_lib.tree import build_tree, render_tree
from luminos_lib.filetypes import classify_files, summarize_categories
from luminos_lib.code import detect_languages, find_large_files
from luminos_lib.recency import find_recent_files
from luminos_lib.disk import get_disk_usage, top_directories
from luminos_lib.watch import watch_loop
from luminos_lib.report import format_report
def _progress(label):
"""Return (on_file, finish) for in-place per-file progress on stderr.
on_file(path) overwrites the current line with the label and truncated path.
finish() finalises the line with a newline.
"""
cols = shutil.get_terminal_size((80, 20)).columns
prefix = f" [scan] {label}... "
available = max(cols - len(prefix), 10)
def on_file(path):
rel = os.path.relpath(path)
if len(rel) > available:
rel = "..." + rel[-(available - 3):]
print(f"\r{prefix}{rel}\033[K", end="", file=sys.stderr, flush=True)
def finish():
print(f"\r{prefix}done\033[K", file=sys.stderr, flush=True)
return on_file, finish
def scan(target, depth=3, show_hidden=False):
"""Run all analyses on the target directory and return a report dict."""
report = {}
print(f" [scan] Building directory tree (depth={depth})...", file=sys.stderr)
tree = build_tree(target, max_depth=depth, show_hidden=show_hidden)
report["tree"] = tree
report["tree_rendered"] = render_tree(tree)
on_file, finish = _progress("Classifying files")
classified = classify_files(target, show_hidden=show_hidden, on_file=on_file)
finish()
report["file_categories"] = summarize_categories(classified)
report["classified_files"] = classified
on_file, finish = _progress("Counting lines")
languages, loc = detect_languages(classified, on_file=on_file)
finish()
report["languages"] = languages
report["lines_of_code"] = loc
on_file, finish = _progress("Checking for large files")
report["large_files"] = find_large_files(classified, on_file=on_file)
finish()
print(" [scan] Finding recently modified files...", file=sys.stderr)
report["recent_files"] = find_recent_files(target, show_hidden=show_hidden)
print(" [scan] Calculating disk usage...", file=sys.stderr)
usage = get_disk_usage(target, show_hidden=show_hidden)
report["disk_usage"] = usage
report["top_directories"] = top_directories(usage, n=5)
print(" [scan] Base scan complete.", file=sys.stderr)
return report
def main():
parser = argparse.ArgumentParser(
prog="luminos",
description="Luminos — file system intelligence tool. "
"Explores a directory and produces a reconnaissance report.",
)
parser.add_argument("target", nargs="?", help="Target directory to analyze")
parser.add_argument("-d", "--depth", type=int, default=3,
help="Maximum tree depth (default: 3)")
parser.add_argument("-a", "--all", action="store_true",
help="Include hidden files and directories")
parser.add_argument("--json", action="store_true", dest="json_output",
help="Output report as JSON")
parser.add_argument("-o", "--output", metavar="FILE",
help="Write report to a file")
parser.add_argument("--ai", action="store_true",
help="Use Claude AI to analyze directory purpose "
"(requires ANTHROPIC_API_KEY)")
parser.add_argument("--watch", action="store_true",
help="Re-scan every 30 seconds and show diffs")
parser.add_argument("--clear-cache", action="store_true",
help="Clear the AI investigation cache (/tmp/luminos/)")
parser.add_argument("--fresh", action="store_true",
help="Force a new AI investigation (ignore cached results)")
parser.add_argument("--install-extras", action="store_true",
help="Show status of optional AI dependencies")
args = parser.parse_args()
# --install-extras: show package status and exit
if args.install_extras:
from luminos_lib.capabilities import print_status
print_status()
return
# --clear-cache: wipe /tmp/luminos/ (lazy import to avoid AI deps)
if args.clear_cache:
from luminos_lib.capabilities import clear_cache
clear_cache()
if not args.target:
return
if not args.target:
parser.error("the following arguments are required: target")
target = os.path.abspath(args.target)
if not os.path.isdir(target):
print(f"Error: '{args.target}' is not a directory or does not exist.",
file=sys.stderr)
sys.exit(1)
if args.watch:
watch_loop(target, depth=args.depth, show_hidden=args.all,
json_output=args.json_output)
return
report = scan(target, depth=args.depth, show_hidden=args.all)
flags = []
if args.ai:
from luminos_lib.ai import analyze_directory
brief, detailed, flags = analyze_directory(report, target, fresh=args.fresh)
report["ai_brief"] = brief
report["ai_detailed"] = detailed
report["flags"] = flags
if args.json_output:
output = json.dumps(report, indent=2, default=str)
else:
output = format_report(report, target, flags=flags)
if args.output:
try:
with open(args.output, "w") as f:
f.write(output + "\n")
print(f"Report written to {args.output}")
except OSError as e:
print(f"Error writing to '{args.output}': {e}", file=sys.stderr)
sys.exit(1)
else:
print(output)
if __name__ == "__main__":
main()