| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858 |
- #!/usr/bin/env bash
- #############################################################################
- # Auto-Detect Components Script v2.0.0
- # Scans .opencode directory for new components not in registry
- # Validates existing entries, fixes typos, removes deleted components
- # Performs security checks on component files
- #############################################################################
- set -e
- # Colors
- RED='\033[0;31m'
- GREEN='\033[0;32m'
- YELLOW='\033[1;33m'
- BLUE='\033[0;34m'
- CYAN='\033[0;36m'
- MAGENTA='\033[0;35m'
- BOLD='\033[1m'
- NC='\033[0m'
- # Configuration
- REGISTRY_FILE="registry.json"
- REPO_ROOT="$(cd "$(dirname "${BASH_SOURCE[0]}")/../.." && pwd)"
- AUTO_ADD=false
- DRY_RUN=false
- VALIDATE_EXISTING=true
- SECURITY_CHECK=true
- # Arrays to store components
- declare -a NEW_COMPONENTS
- declare -a FIXED_COMPONENTS
- declare -a REMOVED_COMPONENTS
- declare -a SECURITY_ISSUES
- # Counters
- TOTAL_FIXED=0
- TOTAL_REMOVED=0
- TOTAL_SECURITY_ISSUES=0
- #############################################################################
- # Utility Functions
- #############################################################################
- print_header() {
- echo -e "${CYAN}${BOLD}"
- echo "╔════════════════════════════════════════════════════════════════╗"
- echo "║ ║"
- echo "║ Auto-Detect Components v2.0.0 ║"
- echo "║ Enhanced with Security & Validation ║"
- echo "║ ║"
- echo "╚════════════════════════════════════════════════════════════════╝"
- echo -e "${NC}"
- }
- print_success() {
- echo -e "${GREEN}✓${NC} $1"
- }
- print_error() {
- echo -e "${RED}✗${NC} $1"
- }
- print_warning() {
- echo -e "${YELLOW}⚠${NC} $1"
- }
- print_info() {
- echo -e "${BLUE}ℹ${NC} $1"
- }
- print_security() {
- echo -e "${MAGENTA}🔒${NC} $1"
- }
- usage() {
- echo "Usage: $0 [OPTIONS]"
- echo ""
- echo "Options:"
- echo " -a, --auto-add Automatically add new components to registry"
- echo " -d, --dry-run Show what would be changed without modifying registry"
- echo " -s, --skip-validation Skip validation of existing registry entries"
- echo " -n, --no-security Skip security checks on component files"
- echo " -h, --help Show this help message"
- echo ""
- echo "Features:"
- echo " • Detects new components in .opencode directory"
- echo " • Validates existing registry entries"
- echo " • Auto-fixes typos and wrong paths"
- echo " • Removes entries for deleted components"
- echo " • Performs security checks (permissions, secrets, path validation)"
- echo ""
- exit 0
- }
- #############################################################################
- # Security Functions
- #############################################################################
- check_file_security() {
- local file=$1
- local issues=()
-
- # For markdown files, be less strict (they contain examples and documentation)
- if [[ "$file" == *.md ]]; then
- # Only check for executable permissions on markdown
- if [ -x "$file" ]; then
- issues+=("Markdown file should not be executable")
- fi
-
- # Check for actual secrets (not examples) - very specific patterns
- # Look for real API keys like sk-proj-xxxxx or ghp_xxxxx
- if grep -qE '(sk-proj-[a-zA-Z0-9]{40,}|ghp_[a-zA-Z0-9]{36,}|xox[baprs]-[a-zA-Z0-9-]{10,})' "$file" 2>/dev/null; then
- issues+=("Potential real API key detected")
- fi
- else
- # For non-markdown files, be more strict
- # Check file permissions (should not be world-writable)
- if [ -w "$file" ] && [ "$(stat -f '%A' "$file" 2>/dev/null || stat -c '%a' "$file" 2>/dev/null)" -gt 664 ]; then
- issues+=("File has overly permissive permissions")
- fi
-
- # Check for potential secrets
- if grep -qiE '(password|secret|api[_-]?key|token|credential|private[_-]?key).*[:=].*[a-zA-Z0-9]{20,}' "$file" 2>/dev/null; then
- issues+=("Potential hardcoded secrets detected")
- fi
- fi
-
- # Return issues
- if [ ${#issues[@]} -gt 0 ]; then
- printf '%s\n' "${issues[@]}"
- return 1
- fi
- return 0
- }
- run_security_checks() {
- if [ "$SECURITY_CHECK" = false ]; then
- return 0
- fi
-
- print_info "Running security checks..."
- echo ""
-
- local categories=("agent" "command" "tool" "plugin" "context")
-
- for category in "${categories[@]}"; do
- local category_dir="$REPO_ROOT/.opencode/$category"
-
- if [ ! -d "$category_dir" ]; then
- continue
- fi
-
- while IFS= read -r file; do
- local rel_path="${file#$REPO_ROOT/}"
-
- # Skip excluded directories
- if [[ "$rel_path" == *"/node_modules/"* ]] || \
- [[ "$rel_path" == *"/tests/"* ]] || \
- [[ "$rel_path" == *"/docs/"* ]]; then
- continue
- fi
-
- # Check security
- local security_output
- if ! security_output=$(check_file_security "$file"); then
- TOTAL_SECURITY_ISSUES=$((TOTAL_SECURITY_ISSUES + 1))
- SECURITY_ISSUES+=("${rel_path}|${security_output}")
- print_security "Security issue in: ${rel_path}"
- while IFS= read -r issue; do
- echo " - ${issue}"
- done <<< "$security_output"
- echo ""
- fi
- done < <(find "$category_dir" -type f -name "*.md" 2>/dev/null)
- done
-
- if [ $TOTAL_SECURITY_ISSUES -eq 0 ]; then
- print_success "No security issues found"
- echo ""
- fi
- }
- #############################################################################
- # Path Validation and Fixing
- #############################################################################
- find_similar_path() {
- local wrong_path=$1
-
- # Get directory and filename
- local dir
- dir=$(dirname "$wrong_path")
- local filename
- filename=$(basename "$wrong_path")
-
- # First, try to find exact filename match in category subdirectories
- # e.g., .opencode/agent/opencoder.md → .opencode/agent/core/opencoder.md
- local base_dir
- base_dir=$(echo "$dir" | cut -d'/' -f1-2) # e.g., .opencode/agent
-
- if [ -d "$REPO_ROOT/$base_dir" ]; then
- # Search recursively in the base directory for exact filename match
- while IFS= read -r candidate; do
- local candidate_rel
- candidate_rel="${candidate#$REPO_ROOT/}"
- local candidate_name
- candidate_name=$(basename "$candidate")
-
- # Exact filename match
- if [[ "$candidate_name" == "$filename" ]]; then
- echo "$candidate_rel"
- return 0
- fi
- done < <(find "$REPO_ROOT/$base_dir" -type f -name "$filename" 2>/dev/null)
- fi
-
- # Fallback: search for similar names in the entire .opencode directory
- local search_dirs=("$REPO_ROOT/$dir" "$REPO_ROOT/.opencode")
-
- for search_dir in "${search_dirs[@]}"; do
- if [ ! -d "$search_dir" ]; then
- continue
- fi
-
- # Find files with similar names
- while IFS= read -r candidate; do
- local candidate_rel
- candidate_rel="${candidate#$REPO_ROOT/}"
- local candidate_name
- candidate_name=$(basename "$candidate")
-
- # Simple similarity check
- if [[ "$candidate_name" == *"$filename"* ]] || [[ "$filename" == *"$candidate_name"* ]]; then
- echo "$candidate_rel"
- return 0
- fi
- done < <(find "$search_dir" -type f -name "*.md" 2>/dev/null)
- done
-
- return 1
- }
- validate_existing_entries() {
- if [ "$VALIDATE_EXISTING" = false ]; then
- return 0
- fi
-
- print_info "Validating existing registry entries..."
- echo ""
-
- # Get all component types from registry
- local component_types
- component_types=$(jq -r '.components | keys[]' "$REGISTRY_FILE" 2>/dev/null)
-
- while IFS= read -r comp_type; do
- # Get all components of this type
- local count
- count=$(jq -r ".components.${comp_type} | length" "$REGISTRY_FILE" 2>/dev/null)
-
- for ((i=0; i<count; i++)); do
- local id
- id=$(jq -r ".components.${comp_type}[$i].id" "$REGISTRY_FILE" 2>/dev/null)
- local name
- name=$(jq -r ".components.${comp_type}[$i].name" "$REGISTRY_FILE" 2>/dev/null)
- local path
- path=$(jq -r ".components.${comp_type}[$i].path" "$REGISTRY_FILE" 2>/dev/null)
-
- # Skip if path is null or empty
- if [ -z "$path" ] || [ "$path" = "null" ]; then
- continue
- fi
-
- local full_path="$REPO_ROOT/$path"
-
- # Check if file exists
- if [ ! -f "$full_path" ]; then
- print_warning "Component file not found: ${name} (${path})"
-
- # Try to find similar path
- local similar_path
- if similar_path=$(find_similar_path "$path"); then
- print_info "Found similar path: ${similar_path}"
-
- if [ "$AUTO_ADD" = true ] && [ "$DRY_RUN" = false ]; then
- fix_component_path "$comp_type" "$i" "$id" "$name" "$path" "$similar_path"
- else
- FIXED_COMPONENTS+=("${comp_type}|${i}|${id}|${name}|${path}|${similar_path}")
- echo " Would fix: ${path} → ${similar_path}"
- fi
- else
- # No similar path found, mark for removal
- if [ "$AUTO_ADD" = true ] && [ "$DRY_RUN" = false ]; then
- remove_component_from_registry "$comp_type" "$id" "$name" "$path"
- else
- REMOVED_COMPONENTS+=("${comp_type}|${id}|${name}|${path}")
- echo " Would remove: ${name} (deleted)"
- fi
- fi
- echo ""
- fi
- done
- done <<< "$component_types"
- }
- fix_component_path() {
- local comp_type=$1
- local index=$2
- local id=$3
- local name=$4
- local old_path=$5
- local new_path=$6
-
- local temp_file="${REGISTRY_FILE}.tmp"
-
- jq --arg type "$comp_type" \
- --argjson idx "$index" \
- --arg newpath "$new_path" \
- ".components[\$type][\$idx].path = \$newpath" \
- "$REGISTRY_FILE" > "$temp_file"
-
- if [ $? -eq 0 ]; then
- mv "$temp_file" "$REGISTRY_FILE"
- print_success "Fixed path for ${name}: ${old_path} → ${new_path}"
- TOTAL_FIXED=$((TOTAL_FIXED + 1))
- else
- print_error "Failed to fix path for ${name}"
- rm -f "$temp_file"
- return 1
- fi
- }
- remove_component_from_registry() {
- local comp_type=$1
- local id=$2
- local name=$3
- local path=$4
-
- local temp_file="${REGISTRY_FILE}.tmp"
-
- jq --arg type "$comp_type" \
- --arg id "$id" \
- ".components[\$type] = [.components[\$type][] | select(.id != \$id)]" \
- "$REGISTRY_FILE" > "$temp_file"
-
- if [ $? -eq 0 ]; then
- mv "$temp_file" "$REGISTRY_FILE"
- print_success "Removed deleted component: ${name}"
- TOTAL_REMOVED=$((TOTAL_REMOVED + 1))
- else
- print_error "Failed to remove component: ${name}"
- rm -f "$temp_file"
- return 1
- fi
- }
- #############################################################################
- # Component Detection
- #############################################################################
- extract_metadata_from_file() {
- local file=$1
- local id=""
- local name=""
- local description=""
- local tags=""
- local dependencies=""
-
- # Try to extract from frontmatter (YAML)
- if grep -q "^---$" "$file" 2>/dev/null; then
- # Extract description from frontmatter
- description=$(sed -n '/^---$/,/^---$/p' "$file" | grep "^description:" | sed 's/^description: *//; s/^"//; s/"$//' | head -1)
-
- # Extract tags from frontmatter (YAML array format)
- # Handles both: tags: [tag1, tag2] and multi-line format
- local in_tags=false
- local frontmatter
- frontmatter=$(sed -n '/^---$/,/^---$/p' "$file")
- while IFS= read -r line; do
- if [[ "$line" =~ ^tags: ]]; then
- # Inline array format: tags: [tag1, tag2]
- if [[ "$line" =~ \[.*\] ]]; then
- tags=$(echo "$line" | sed 's/^tags: *\[//; s/\].*//; s/ //g')
- else
- in_tags=true
- fi
- elif [[ "$in_tags" == true ]]; then
- # Multi-line array format
- if [[ "$line" =~ ^---$ ]]; then
- # End of frontmatter
- in_tags=false
- elif [[ "$line" =~ ^[[:space:]]*- ]]; then
- local tag
- tag=$(echo "$line" | sed 's/^[[:space:]]*- *//')
- if [ -z "$tags" ]; then
- tags="$tag"
- else
- tags="$tags,$tag"
- fi
- elif [[ ! "$line" =~ ^[[:space:]] ]]; then
- in_tags=false
- fi
- fi
- done <<< "$frontmatter"
-
- # Extract dependencies from frontmatter (similar to tags)
- # Handles: dependencies: [dep1, dep2] or multi-line format
- local in_deps=false
- while IFS= read -r line; do
- if [[ "$line" =~ ^dependencies: ]]; then
- # Inline array format
- if [[ "$line" =~ \[.*\] ]]; then
- dependencies=$(echo "$line" | sed 's/^dependencies: *\[//; s/\].*//; s/ //g; s/"//g; s/'"'"'//g')
- else
- in_deps=true
- fi
- elif [[ "$in_deps" == true ]]; then
- # Multi-line array format
- if [[ "$line" =~ ^---$ ]]; then
- # End of frontmatter
- in_deps=false
- elif [[ "$line" =~ ^[[:space:]]*- ]]; then
- # Extract dependency (remove leading dash and whitespace)
- local dep
- dep=$(echo "$line" | sed 's/^[[:space:]]*- *//' | sed 's/"//g; s/'"'"'//g' | sed 's/#.*//' | xargs)
- # Skip empty lines and comments
- if [ -n "$dep" ] && [[ ! "$dep" =~ ^# ]]; then
- if [ -z "$dependencies" ]; then
- dependencies="$dep"
- else
- dependencies="$dependencies,$dep"
- fi
- fi
- elif [[ "$line" =~ ^[[:space:]]*# ]] || [[ -z "$line" ]]; then
- # Skip comments and empty lines within dependencies
- continue
- elif [[ ! "$line" =~ ^[[:space:]] ]]; then
- # Non-indented line means we've moved to a new field
- in_deps=false
- fi
- fi
- done <<< "$frontmatter"
- fi
-
- # If no description in frontmatter, try to get from first heading or paragraph
- if [ -z "$description" ]; then
- description=$(grep -m 1 "^# " "$file" | sed 's/^# //' || echo "")
- fi
-
- # Generate ID from filename
- local filename
- filename=$(basename "$file" .md)
- id=$(echo "$filename" | tr '[:upper:]' '[:lower:]' | tr ' ' '-')
-
- # Generate name from filename (capitalize words)
- name=$(echo "$filename" | sed 's/-/ /g' | awk '{for(i=1;i<=NF;i++) $i=toupper(substr($i,1,1)) tolower(substr($i,2))}1')
-
- echo "${id}|${name}|${description}|${tags}|${dependencies}"
- }
- detect_component_type() {
- local path=$1
-
- # Handle category-based paths (e.g., .opencode/agent/core/openagent.md)
- # and flat paths (e.g., .opencode/agent/openagent.md)
- if [[ "$path" == *"/agent/subagents/"* ]]; then
- echo "subagent"
- elif [[ "$path" == *"/agent/"* ]]; then
- echo "agent"
- elif [[ "$path" == *"/command/"* ]]; then
- echo "command"
- elif [[ "$path" == *"/tool/"* ]]; then
- echo "tool"
- elif [[ "$path" == *"/plugin/"* ]]; then
- echo "plugin"
- elif [[ "$path" == *"/context/"* ]]; then
- echo "context"
- else
- echo "unknown"
- fi
- }
- extract_category_from_path() {
- local path=$1
-
- # Extract category from path like .opencode/agent/core/openagent.md → core
- # or .opencode/agent/subagents/code/tester.md → code (for subagents)
- if [[ "$path" == *"/agent/subagents/"* ]]; then
- # For subagents: .opencode/agent/subagents/code/tester.md → code
- echo "$path" | sed -E 's|.*/agent/subagents/([^/]+)/.*|\1|'
- elif [[ "$path" == *"/agent/"* ]]; then
- # For agents: .opencode/agent/core/openagent.md → core
- # Check if there's a category subdirectory
- local category
- category=$(echo "$path" | sed -E 's|.*/agent/([^/]+)/.*|\1|')
- # If category is the filename, it's a flat structure (no category)
- if [[ "$category" == *.md ]]; then
- echo "standard"
- else
- echo "$category"
- fi
- else
- echo "standard"
- fi
- }
- get_registry_key() {
- local type=$1
- case "$type" in
- config) echo "config" ;;
- *) echo "${type}s" ;;
- esac
- }
- scan_for_new_components() {
- print_info "Scanning for new components..."
- echo ""
-
- # Get all paths from registry
- local registry_paths
- registry_paths=$(jq -r '.components | to_entries[] | .value[] | .path' "$REGISTRY_FILE" 2>/dev/null | sort -u)
-
- # Scan .opencode directory
- local categories=("agent" "command" "tool" "plugin" "context")
-
- for category in "${categories[@]}"; do
- local category_dir="$REPO_ROOT/.opencode/$category"
-
- if [ ! -d "$category_dir" ]; then
- continue
- fi
-
- # Find all .md files recursively (excluding node_modules, tests, docs, templates)
- while IFS= read -r file; do
- local rel_path="${file#$REPO_ROOT/}"
-
- # Skip symlinks (backward compatibility links)
- if [ -L "$file" ]; then
- continue
- fi
-
- # Skip node_modules, tests, docs, templates
- if [[ "$rel_path" == *"/node_modules/"* ]] || \
- [[ "$rel_path" == *"/tests/"* ]] || \
- [[ "$rel_path" == *"/docs/"* ]] || \
- [[ "$rel_path" == *"/template"* ]] || \
- [[ "$rel_path" == *"README.md" ]] || \
- [[ "$rel_path" == *"index.md" ]]; then
- continue
- fi
-
- # Check if this path is in registry
- if ! echo "$registry_paths" | grep -q "^${rel_path}$"; then
- # Extract metadata
- local metadata
- metadata=$(extract_metadata_from_file "$file")
- IFS='|' read -r id name description tags dependencies <<< "$metadata"
-
- # Detect component type
- local comp_type
- comp_type=$(detect_component_type "$rel_path")
-
- # Extract category from path
- local comp_category
- comp_category=$(extract_category_from_path "$rel_path")
-
- if [ "$comp_type" != "unknown" ]; then
- NEW_COMPONENTS+=("${comp_type}|${id}|${name}|${description}|${rel_path}|${comp_category}|${tags}|${dependencies}")
- print_warning "New ${comp_type}: ${name} (${id})"
- echo " Path: ${rel_path}"
- echo " Category: ${comp_category}"
- [ -n "$description" ] && echo " Description: ${description}"
- [ -n "$tags" ] && echo " Tags: ${tags}"
- [ -n "$dependencies" ] && echo " Dependencies: ${dependencies}"
-
- # Check dependencies
- if [ -n "$dependencies" ]; then
- check_dependencies "$dependencies" "$name"
- fi
-
- echo ""
- fi
- fi
- done < <(find "$category_dir" -type f -name "*.md" 2>/dev/null)
- done
- }
- check_dependencies() {
- local deps_str=$1
- # local component_name=$2 # Unused
-
- if [ -z "$deps_str" ]; then
- return 0
- fi
-
- # Split dependencies by comma
- IFS=',' read -ra deps <<< "$deps_str"
- for dep in "${deps[@]}"; do
- dep=$(echo "$dep" | sed 's/^[[:space:]]*//;s/[[:space:]]*$//')
-
- if [ -z "$dep" ]; then
- continue
- fi
-
- # Parse dependency: type:id
- if [[ ! "$dep" =~ ^([^:]+):(.+)$ ]]; then
- echo " ⚠ Invalid dependency format: ${dep} (expected type:id)"
- continue
- fi
-
- local dep_type="${BASH_REMATCH[1]}"
- local dep_id="${BASH_REMATCH[2]}"
-
- # Map to registry category
- local category=""
- case "$dep_type" in
- agent) category="agents" ;;
- subagent) category="subagents" ;;
- command) category="commands" ;;
- tool) category="tools" ;;
- plugin) category="plugins" ;;
- context) category="contexts" ;;
- config) category="config" ;;
- *)
- echo " ⚠ Unknown dependency type: ${dep}"
- continue
- ;;
- esac
-
- # Check if exists in registry
- local exists
- exists=$(jq -r ".components.${category}[]? | select(.id == \"${dep_id}\") | .id" "$REGISTRY_FILE" 2>/dev/null)
-
- if [ -z "$exists" ]; then
- echo " ⚠ Dependency not found in registry: ${dep}"
- fi
- done
- }
- add_component_to_registry() {
- local comp_type=$1
- local id=$2
- local name=$3
- local description=$4
- local path=$5
- local comp_category=${6:-"standard"}
- local tags_str=${7:-""}
- local deps_str=${8:-""}
-
- # Default description if empty
- if [ -z "$description" ]; then
- description="Component: ${name}"
- fi
-
- # Escape quotes and special characters in description
- description=$(echo "$description" | sed 's/"/\\"/g' | sed "s/'/\\'/g")
-
- # Convert comma-separated strings to JSON arrays
- local tags_json="[]"
- if [ -n "$tags_str" ]; then
- tags_json=$(echo "$tags_str" | awk -F',' '{printf "["; for(i=1;i<=NF;i++) {gsub(/^[ \t]+|[ \t]+$/, "", $i); printf "\"%s\"", $i; if(i<NF) printf ","}; printf "]"}')
- fi
-
- local deps_json="[]"
- if [ -n "$deps_str" ]; then
- deps_json=$(echo "$deps_str" | awk -F',' '{printf "["; for(i=1;i<=NF;i++) {gsub(/^[ \t]+|[ \t]+$/, "", $i); printf "\"%s\"", $i; if(i<NF) printf ","}; printf "]"}')
- fi
-
- # Get registry key (agents, subagents, commands, etc.)
- local registry_key
- registry_key=$(get_registry_key "$comp_type")
-
- # Use jq to properly construct JSON (avoids escaping issues)
- local temp_file="${REGISTRY_FILE}.tmp"
- jq --arg id "$id" \
- --arg name "$name" \
- --arg type "$comp_type" \
- --arg path "$path" \
- --arg desc "$description" \
- --arg cat "$comp_category" \
- --argjson tags "$tags_json" \
- --argjson deps "$deps_json" \
- ".components.${registry_key} += [{
- \"id\": \$id,
- \"name\": \$name,
- \"type\": \$type,
- \"path\": \$path,
- \"description\": \$desc,
- \"tags\": \$tags,
- \"dependencies\": \$deps,
- \"category\": \$cat
- }]" "$REGISTRY_FILE" > "$temp_file"
-
- if [ $? -eq 0 ]; then
- mv "$temp_file" "$REGISTRY_FILE"
- print_success "Added ${comp_type}: ${name} (category: ${comp_category})"
- else
- print_error "Failed to add ${comp_type}: ${name}"
- rm -f "$temp_file"
- return 1
- fi
- }
- #############################################################################
- # Main
- #############################################################################
- main() {
- # Parse arguments
- while [ $# -gt 0 ]; do
- case "$1" in
- -a|--auto-add)
- AUTO_ADD=true
- shift
- ;;
- -d|--dry-run)
- DRY_RUN=true
- shift
- ;;
- -s|--skip-validation)
- VALIDATE_EXISTING=false
- shift
- ;;
- -n|--no-security)
- SECURITY_CHECK=false
- shift
- ;;
- -h|--help)
- usage
- ;;
- *)
- echo "Unknown option: $1"
- usage
- ;;
- esac
- done
-
- print_header
-
- # Check dependencies
- if ! command -v jq &> /dev/null; then
- print_error "jq is required but not installed"
- exit 1
- fi
-
- # Validate registry file
- if [ ! -f "$REGISTRY_FILE" ]; then
- print_error "Registry file not found: $REGISTRY_FILE"
- exit 1
- fi
-
- if ! jq empty "$REGISTRY_FILE" 2>/dev/null; then
- print_error "Registry file is not valid JSON"
- exit 1
- fi
-
- # Run security checks
- run_security_checks
-
- # Validate existing entries (fixes and removals)
- validate_existing_entries
-
- # Scan for new components
- scan_for_new_components
-
- # Summary
- echo ""
- echo -e "${BOLD}═══════════════════════════════════════════════════════════════${NC}"
- echo -e "${BOLD}Summary${NC}"
- echo -e "${BOLD}═══════════════════════════════════════════════════════════════${NC}"
- echo ""
-
- # Display counts
- echo -e "Security Issues: ${MAGENTA}${TOTAL_SECURITY_ISSUES}${NC}"
- echo -e "Fixed Paths: ${GREEN}${TOTAL_FIXED}${NC}"
- echo -e "Removed Components: ${RED}${TOTAL_REMOVED}${NC}"
- echo -e "New Components: ${YELLOW}${#NEW_COMPONENTS[@]}${NC}"
- echo ""
-
- # Show pending fixes if in dry-run mode
- if [ ${#FIXED_COMPONENTS[@]} -gt 0 ] && [ "$DRY_RUN" = true ]; then
- echo -e "${BOLD}Pending Path Fixes:${NC}"
- for entry in "${FIXED_COMPONENTS[@]}"; do
- IFS='|' read -r comp_type index id name old_path new_path <<< "$entry"
- echo " • ${name}: ${old_path} → ${new_path}"
- done
- echo ""
- fi
-
- # Show pending removals if in dry-run mode
- if [ ${#REMOVED_COMPONENTS[@]} -gt 0 ] && [ "$DRY_RUN" = true ]; then
- echo -e "${BOLD}Pending Removals:${NC}"
- for entry in "${REMOVED_COMPONENTS[@]}"; do
- IFS='|' read -r comp_type id name path <<< "$entry"
- echo " • ${name} (${path})"
- done
- echo ""
- fi
-
- # Check if everything is up to date
- if [ ${#NEW_COMPONENTS[@]} -eq 0 ] && \
- [ ${#FIXED_COMPONENTS[@]} -eq 0 ] && \
- [ ${#REMOVED_COMPONENTS[@]} -eq 0 ] && \
- [ $TOTAL_FIXED -eq 0 ] && \
- [ $TOTAL_REMOVED -eq 0 ]; then
- print_success "Registry is up to date!"
-
- if [ $TOTAL_SECURITY_ISSUES -gt 0 ]; then
- echo ""
- print_warning "Please review and fix the ${TOTAL_SECURITY_ISSUES} security issue(s) found"
- fi
-
- exit 0
- fi
-
- # Add components if auto-add is enabled
- if [ "$AUTO_ADD" = true ] && [ "$DRY_RUN" = false ]; then
- if [ ${#NEW_COMPONENTS[@]} -gt 0 ]; then
- print_info "Adding new components to registry..."
- echo ""
-
- local added=0
- for entry in "${NEW_COMPONENTS[@]}"; do
- IFS='|' read -r comp_type id name description path comp_category tags dependencies <<< "$entry"
- if add_component_to_registry "$comp_type" "$id" "$name" "$description" "$path" "$comp_category" "$tags" "$dependencies"; then
- added=$((added + 1))
- fi
- done
-
- echo ""
- print_success "Added ${added} component(s) to registry"
- fi
-
- # Update timestamp
- jq '.metadata.lastUpdated = (now | strftime("%Y-%m-%d"))' "$REGISTRY_FILE" > "${REGISTRY_FILE}.tmp"
- mv "${REGISTRY_FILE}.tmp" "$REGISTRY_FILE"
-
- elif [ "$DRY_RUN" = true ]; then
- print_info "Dry run mode - no changes made to registry"
- echo ""
- echo "Run without --dry-run to apply these changes"
-
- else
- print_info "Run with --auto-add to apply these changes to registry"
- echo ""
- echo "Or manually update registry.json"
- fi
-
- # Final security warning
- if [ $TOTAL_SECURITY_ISSUES -gt 0 ]; then
- echo ""
- print_warning "⚠️ ${TOTAL_SECURITY_ISSUES} security issue(s) require attention"
- fi
-
- exit 0
- }
- main "$@"
|