Files
Antigravity bd495be965
All checks were successful
Deploy to Production / Build and Deploy (push) Successful in 12s
feat: design system overhaul — sidebar, AI chats, settings, brainstorm, color cleanup
- Sidebar: dynamic brand-accent colors, brainstorm section restyled
- AI chat general: popup panel with expand/collapse, hides when contextual AI open
- AI chat contextual: tabs reordered (Actions first), X close button, height fix
- Settings: all tabs restyled, 6 new color presets (sage, terracotta, iron, etc.)
- Global color cleanup: emerald/orange hardcoded → brand-accent dynamic
- Brainstorm page: orange → brand-accent throughout
- PageEntry animation component added to key pages
- Floating AI button: bg-brand-accent instead of hardcoded black
- i18n: all 15 locales updated with new AI/billing keys
- Billing: freemium quota tracking, BYOK, stripe subscription scaffolding
- Admin: integrated into new design
- AGENTS.md + CLAUDE.md project rules added
2026-05-16 12:59:30 +00:00

249 lines
8.8 KiB
Python

#!/usr/bin/env python3
# /// script
# requires-python = ">=3.10"
# dependencies = []
# ///
"""Produce structured diff between original and transformed lyrics.
Compares two versions of lyrics and categorizes changes by type (added,
removed, modified) and tracks which sections they fall in.
Usage:
python lyrics-diff.py --original orig.txt --transformed trans.txt [options]
# Compare two files
python lyrics-diff.py --original orig.txt --transformed trans.txt
# Compare two text strings
python lyrics-diff.py --original-text "old lyrics" --transformed-text "new lyrics"
# Output to file
python lyrics-diff.py --original orig.txt --transformed trans.txt -o diff.json
"""
import argparse
import difflib
import json
import re
import sys
from datetime import datetime, timezone
from pathlib import Path
SCRIPT_NAME = "lyrics-diff"
VERSION = "1.0.0"
def get_section_at_line(lines: list[str], line_idx: int) -> str:
"""Determine which section a given line index falls in."""
current_section = "(no section)"
for i in range(line_idx + 1):
if i < len(lines):
stripped = lines[i].strip()
tag_match = re.match(r'^\[([^\]:]+)\]$', stripped)
if tag_match:
current_section = tag_match.group(1).strip()
return current_section
def compute_diff(original: str, transformed: str) -> dict:
"""Compute structured diff between original and transformed lyrics."""
orig_lines = original.split('\n')
trans_lines = transformed.split('\n')
matcher = difflib.SequenceMatcher(None, orig_lines, trans_lines)
changes = []
sections_affected = set()
for tag, i1, i2, j1, j2 in matcher.get_opcodes():
if tag == 'equal':
continue
elif tag == 'replace':
# Modified lines
max_len = max(i2 - i1, j2 - j1)
for k in range(max_len):
orig_idx = i1 + k if k < (i2 - i1) else None
trans_idx = j1 + k if k < (j2 - j1) else None
if orig_idx is not None and trans_idx is not None:
section = get_section_at_line(orig_lines, orig_idx)
sections_affected.add(section)
changes.append({
"type": "modified",
"section": section,
"line": orig_idx + 1,
"original": orig_lines[orig_idx],
"transformed": trans_lines[trans_idx]
})
elif orig_idx is not None:
section = get_section_at_line(orig_lines, orig_idx)
sections_affected.add(section)
changes.append({
"type": "removed",
"section": section,
"line": orig_idx + 1,
"original": orig_lines[orig_idx],
"transformed": ""
})
elif trans_idx is not None:
section = get_section_at_line(trans_lines, trans_idx)
sections_affected.add(section)
changes.append({
"type": "added",
"section": section,
"line": trans_idx + 1,
"original": "",
"transformed": trans_lines[trans_idx]
})
elif tag == 'delete':
for k in range(i1, i2):
section = get_section_at_line(orig_lines, k)
sections_affected.add(section)
changes.append({
"type": "removed",
"section": section,
"line": k + 1,
"original": orig_lines[k],
"transformed": ""
})
elif tag == 'insert':
for k in range(j1, j2):
section = get_section_at_line(trans_lines, k)
sections_affected.add(section)
changes.append({
"type": "added",
"section": section,
"line": k + 1,
"original": "",
"transformed": trans_lines[k]
})
# Generate unified diff for human-readable output
unified = list(difflib.unified_diff(
orig_lines, trans_lines,
fromfile="original", tofile="transformed",
lineterm=""
))
summary = {
"lines_added": sum(1 for c in changes if c["type"] == "added"),
"lines_removed": sum(1 for c in changes if c["type"] == "removed"),
"lines_modified": sum(1 for c in changes if c["type"] == "modified"),
"sections_affected": sorted(sections_affected)
}
return {
"changes": changes,
"unified_diff": "\n".join(unified),
"summary": summary
}
def build_report(result: dict, skill_path: str = "") -> dict:
"""Build the standard output report."""
total_changes = len(result["changes"])
status = "pass"
if total_changes == 0:
status = "pass"
else:
status = "info"
findings = []
if total_changes == 0:
findings.append({
"severity": "info",
"category": "diff",
"issue": "No differences found between original and transformed lyrics.",
"fix": "Lyrics are identical."
})
severity_counts = {"critical": 0, "high": 0, "medium": 0, "low": 0, "info": 0}
for f in findings:
severity_counts[f["severity"]] = severity_counts.get(f["severity"], 0) + 1
return {
"script": SCRIPT_NAME,
"version": VERSION,
"skill_path": skill_path,
"timestamp": datetime.now(timezone.utc).isoformat(),
"status": status,
"changes": result["changes"],
"unified_diff": result["unified_diff"],
"summary": result["summary"],
"findings": findings,
"finding_counts": {
"total": len(findings),
**severity_counts
}
}
def main():
parser = argparse.ArgumentParser(
description="Produce structured diff between original and transformed lyrics.",
formatter_class=argparse.RawDescriptionHelpFormatter,
epilog="""
Examples:
%(prog)s --original orig.txt --transformed trans.txt
%(prog)s --original-text "old lyrics" --transformed-text "new lyrics"
%(prog)s --original orig.txt --transformed trans.txt -o diff.json --verbose
Exit codes: 0=pass, 1=differences found, 2=error
"""
)
parser.add_argument("file", nargs="?", help="Unused (for pattern consistency)")
parser.add_argument("--original", help="Path to original lyrics file")
parser.add_argument("--transformed", help="Path to transformed lyrics file")
parser.add_argument("--original-text", help="Original lyrics text directly")
parser.add_argument("--transformed-text", help="Transformed lyrics text directly")
parser.add_argument("--text", help="Unused (for pattern consistency)")
parser.add_argument("--stdin", action="store_true", help="Unused (for pattern consistency)")
parser.add_argument("-o", "--output", help="Output file path (defaults to stdout)")
parser.add_argument("--verbose", action="store_true", help="Print diagnostics to stderr")
parser.add_argument("--skill-path", default="", help="Skill path for report context")
args = parser.parse_args()
original = ""
transformed = ""
if args.original_text and args.transformed_text:
original = args.original_text.replace('\\n', '\n')
transformed = args.transformed_text.replace('\\n', '\n')
elif args.original and args.transformed:
orig_path = Path(args.original)
trans_path = Path(args.transformed)
if not orig_path.exists():
print(f"Error: File not found: {args.original}", file=sys.stderr)
sys.exit(2)
if not trans_path.exists():
print(f"Error: File not found: {args.transformed}", file=sys.stderr)
sys.exit(2)
original = orig_path.read_text()
transformed = trans_path.read_text()
else:
print("Error: Provide --original and --transformed files, or --original-text and --transformed-text.", file=sys.stderr)
parser.print_help()
sys.exit(2)
if args.verbose:
print(f"Comparing lyrics (original: {len(original)} chars, transformed: {len(transformed)} chars)...", file=sys.stderr)
result = compute_diff(original, transformed)
report = build_report(result, args.skill_path)
output_json = json.dumps(report, indent=2)
if args.output:
Path(args.output).write_text(output_json)
if args.verbose:
print(f"Report written to {args.output}", file=sys.stderr)
else:
print(output_json)
sys.exit(0 if len(result["changes"]) == 0 else 1)
if __name__ == "__main__":
main()