secubox-openwrt/secubox-tools/validate-modules.sh

336 lines
12 KiB
Bash
Executable File

#!/bin/bash
#
# SecuBox Module Validation Script
# Validates RPCD naming, menu paths, and module structure
#
set -e
set -o pipefail
RED='\033[0;31m'
GREEN='\033[0;32m'
YELLOW='\033[1;33m'
NC='\033[0m' # No Color
ERRORS=0
WARNINGS=0
# Helper function to collect all luci-app directories
get_luci_apps() {
find . -maxdepth 1 -type d -name 'luci-app-*' 2>/dev/null
find package/secubox -maxdepth 1 -type d -name 'luci-app-*' 2>/dev/null
}
echo "========================================"
echo "SecuBox Module Validation"
echo "========================================"
echo ""
# Function to print error
error() {
echo -e "${RED}❌ ERROR: $1${NC}"
ERRORS=$((ERRORS + 1))
}
# Function to print warning
warn() {
echo -e "${YELLOW}⚠️ WARNING: $1${NC}"
WARNINGS=$((WARNINGS + 1))
}
# Function to print success
success() {
echo -e "${GREEN}$1${NC}"
}
# Check 1: RPCD script names must match ubus object names
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
echo "1. Validating RPCD script names vs ubus objects"
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
echo ""
while IFS= read -r module_dir; do
[[ ! -d "$module_dir" ]] && continue
module_name=$(basename "$module_dir")
echo "Checking $module_name..."
# Find RPCD script
rpcd_dir="$module_dir/root/usr/libexec/rpcd"
if [ -d "$rpcd_dir" ]; then
rpcd_script=$(find "$rpcd_dir" -type f ! -name "*.md" 2>/dev/null | head -1)
if [ -n "$rpcd_script" ]; then
rpcd_name=$(basename "$rpcd_script")
# Extract ubus object names from JavaScript files
set +e
js_objects=$(find "$module_dir/htdocs" -name "*.js" -type f 2>/dev/null | \
xargs grep -h "object:" 2>/dev/null | \
grep -o "'[^']*'" | sort -u | tr -d "'")
set -e
if [ -n "$js_objects" ]; then
# Check if RPCD script name matches any ubus object
match_found=false
for obj in $js_objects; do
if [ "$rpcd_name" = "$obj" ]; then
match_found=true
success "$module_name: RPCD script '$rpcd_name' matches ubus object '$obj'"
break
fi
done
if [ "$match_found" = false ]; then
error "$module_name: RPCD script '$rpcd_name' does not match any ubus object(s): $js_objects"
echo " → Rename $rpcd_script to one of: $js_objects"
fi
else
warn "$module_name: No ubus object declarations found in JavaScript files"
fi
else
warn "$module_name: No RPCD script found in $rpcd_dir"
fi
else
warn "$module_name: No RPCD directory found"
fi
echo ""
done < <(get_luci_apps)
# Check 2: Menu paths must match actual view file locations
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
echo "2. Validating menu paths vs view file locations"
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
echo ""
while IFS= read -r module_dir; do
[[ ! -d "$module_dir" ]] && continue
module_name=$(basename "$module_dir")
menu_file="$module_dir/root/usr/share/luci/menu.d/${module_name}.json"
if [ -f "$menu_file" ]; then
echo "Checking $module_name menu paths..."
# Extract all view paths from menu JSON
menu_paths=$(grep -o '"path":\s*"[^"]*"' "$menu_file" | cut -d'"' -f4)
for path in $menu_paths; do
# Locate view file anywhere in repo (supports shared menus pointing to other modules)
view_file=$(find . -path "*/htdocs/luci-static/resources/view/${path}.js" -print -quit 2>/dev/null)
if [ -n "$view_file" ] && [ -f "$view_file" ]; then
success "$module_name: Menu path '$path' → file exists at ${view_file#./}"
else
error "$module_name: Menu path '$path' → no view found in repository"
view_dir_guess=$(printf "%s/htdocs/luci-static/resources/view/%s" "$module_dir" "$(dirname "$path")")
if [ -d "$view_dir_guess" ]; then
echo " → Possible files in $(dirname $path):"
find "$view_dir_guess" -name "*.js" -type f | while read -r f; do
echo " - $(basename $f)"
done
fi
fi
done
echo ""
fi
done < <(get_luci_apps)
# Check 3: View files must have corresponding menu entries
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
echo "3. Validating view files have menu entries"
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
echo ""
while IFS= read -r module_dir; do
[[ ! -d "$module_dir" ]] && continue
module_name=$(basename "$module_dir")
view_dir="$module_dir/htdocs/luci-static/resources/view"
menu_file="$module_dir/root/usr/share/luci/menu.d/${module_name}.json"
if [ -d "$view_dir" ] && [ -f "$menu_file" ]; then
echo "Checking $module_name view files..."
# Temporarily disable exit on error for grep checks in loops
set +e
# Find all .js view files
find "$view_dir" -name "*.js" -type f 2>/dev/null | while read -r view_file; do
# Convert file path to menu path
rel_path=$(echo "$view_file" | sed "s|$module_dir/htdocs/luci-static/resources/view/||" | sed 's|.js$||')
# Check if path exists in menu
if grep -q "\"path\":\s*\"$rel_path\"" "$menu_file" 2>/dev/null; then
success "$module_name: View '$rel_path.js' has menu entry"
else
warn "$module_name: View file '$rel_path.js' exists but has no menu entry"
fi
done
set -e
echo ""
fi
done < <(get_luci_apps)
# Check 4: RPCD scripts must be executable
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
echo "4. Validating RPCD script permissions"
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
echo ""
while IFS= read -r module_dir; do
[[ ! -d "$module_dir" ]] && continue
module_name=$(basename "$module_dir")
rpcd_dir="$module_dir/root/usr/libexec/rpcd"
if [ -d "$rpcd_dir" ]; then
find "$rpcd_dir" -type f ! -name "*.md" 2>/dev/null | while read -r script; do
if [ -x "$script" ]; then
success "$module_name: RPCD script $(basename $script) is executable"
else
error "$module_name: RPCD script $(basename $script) is NOT executable"
echo " → Run: chmod +x $script"
fi
done
fi
done < <(get_luci_apps)
echo ""
# Check 5: JSON files must be valid
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
echo "5. Validating JSON syntax"
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
echo ""
while IFS= read -r module_dir; do
[[ ! -d "$module_dir" ]] && continue
module_name=$(basename "$module_dir")
# Check menu JSON
menu_file="$module_dir/root/usr/share/luci/menu.d/${module_name}.json"
if [ -f "$menu_file" ]; then
if python3 -m json.tool "$menu_file" > /dev/null 2>&1; then
success "$module_name: menu.d JSON is valid"
else
error "$module_name: menu.d JSON is INVALID"
fi
fi
# Check ACL JSON
acl_file="$module_dir/root/usr/share/rpcd/acl.d/${module_name}.json"
if [ -f "$acl_file" ]; then
if python3 -m json.tool "$acl_file" > /dev/null 2>&1; then
success "$module_name: acl.d JSON is valid"
else
error "$module_name: acl.d JSON is INVALID"
fi
fi
done < <(get_luci_apps)
echo ""
# Check 6: Verify ubus object naming convention
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
echo "6. Validating ubus object naming convention"
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
echo ""
while IFS= read -r module_dir; do
[[ ! -d "$module_dir" ]] && continue
module_name=$(basename "$module_dir")
# Extract ubus object names from JavaScript
set +e
js_objects=$(find "$module_dir/htdocs" -name "*.js" -type f 2>/dev/null | \
xargs grep -h "object:" 2>/dev/null | \
grep -o "'[^']*'" | sort -u | tr -d "'")
set -e
if [ -n "$js_objects" ]; then
for obj in $js_objects; do
if [[ "$obj" == "service" || "$obj" == "file" ]]; then
continue
fi
# Check if object starts with 'luci.'
if [[ $obj == luci.* ]]; then
success "$module_name: ubus object '$obj' follows naming convention (luci.* prefix)"
else
error "$module_name: ubus object '$obj' does NOT follow naming convention (missing luci. prefix)"
fi
done
fi
done < <(get_luci_apps)
echo ""
# Check 7: htdocs files must have correct permissions (644 for web server)
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
echo "7. Validating htdocs file permissions (CSS/JS must be 644)"
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
echo ""
PERMISSION_ERRORS=0
while IFS= read -r module_dir; do
[[ ! -d "$module_dir" ]] && continue
module_name=$(basename "$module_dir")
htdocs_dir="$module_dir/htdocs"
if [ -d "$htdocs_dir" ]; then
# Check CSS files
while IFS= read -r css_file; do
if [ -n "$css_file" ]; then
perms=$(stat -c "%a" "$css_file" 2>/dev/null)
if [ "$perms" != "644" ]; then
error "$module_name: CSS file has wrong permissions: $css_file ($perms, should be 644)"
echo " → Run: chmod 644 $css_file"
((PERMISSION_ERRORS++))
else
success "$module_name: CSS file has correct permissions (644): $(basename $css_file)"
fi
fi
done < <(find "$htdocs_dir" -name "*.css" -type f 2>/dev/null)
# Check JS files
while IFS= read -r js_file; do
if [ -n "$js_file" ]; then
perms=$(stat -c "%a" "$js_file" 2>/dev/null)
if [ "$perms" != "644" ]; then
error "$module_name: JS file has wrong permissions: $js_file ($perms, should be 644)"
echo " → Run: chmod 644 $js_file"
((PERMISSION_ERRORS++))
else
success "$module_name: JS file has correct permissions (644): $(basename $js_file)"
fi
fi
done < <(find "$htdocs_dir" -name "*.js" -type f 2>/dev/null)
fi
done < <(get_luci_apps)
if [ $PERMISSION_ERRORS -gt 0 ]; then
echo ""
echo -e "${YELLOW}⚠️ To fix all permission errors automatically, run:${NC}"
echo " ./secubox-tools/fix-permissions.sh --local"
fi
echo ""
# Add permission errors to total error count
TOTAL_ERRORS=$((ERRORS + PERMISSION_ERRORS))
# Summary
echo "========================================"
echo "Validation Summary"
echo "========================================"
echo ""
if [ $TOTAL_ERRORS -eq 0 ] && [ $WARNINGS -eq 0 ]; then
echo -e "${GREEN}✓ All checks passed!${NC}"
exit 0
elif [ $TOTAL_ERRORS -eq 0 ]; then
echo -e "${YELLOW}✓ All critical checks passed with $WARNINGS warning(s)${NC}"
exit 0
else
echo -e "${RED}✗ Found $TOTAL_ERRORS error(s) and $WARNINGS warning(s)${NC}"
if [ $PERMISSION_ERRORS -gt 0 ]; then
echo -e "${YELLOW} ($PERMISSION_ERRORS permission error(s))${NC}"
fi
echo ""
echo "Please fix the errors listed above before deploying."
echo "Run: ./secubox-tools/fix-permissions.sh --local"
exit 1
fi