#!/bin/bash

# Test Runner Script for DPDKTokenizer
# This script builds and runs all tests for the DPDK tokenizer utilities

set -e  # Exit on any error

# Colors for output
RED='\033[0;31m'
GREEN='\033[0;32m'
YELLOW='\033[1;33m'
BLUE='\033[0;34m'
NC='\033[0m' # No Color

# Configuration
TEST_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
LOG_LEVEL=${LOG_LEVEL:-3}  # Default log level
HUGE_PAGES=${HUGE_PAGES:-512}  # Default huge pages

# Function to print colored output
print_status() {
    local color=$1
    local message=$2
    echo -e "${color}${message}${NC}"
}

# Function to check DPDK setup
check_dpdk_setup() {
    print_status $BLUE "Checking DPDK setup..."
    
    # Check if DPDK is available
    if ! pkg-config --exists libdpdk; then
        print_status $RED "Error: DPDK not found. Please install DPDK development packages."
        exit 1
    fi
    
    # Check huge pages
    local hugepages=$(cat /proc/sys/vm/nr_hugepages 2>/dev/null || echo "0")
    if [ "$hugepages" -lt "$HUGE_PAGES" ]; then
        print_status $YELLOW "Warning: Insufficient huge pages ($hugepages < $HUGE_PAGES)"
        print_status $YELLOW "Attempting to set huge pages..."
        if [ "$EUID" -eq 0 ]; then
            echo $HUGE_PAGES > /proc/sys/vm/nr_hugepages
        else
            print_status $YELLOW "Run as root or set huge pages manually:"
            print_status $YELLOW "sudo echo $HUGE_PAGES > /proc/sys/vm/nr_hugepages"
        fi
    fi
    
    # Check if we can bind DPDK modules
    if [ ! -d "/sys/module/uio" ] && [ ! -d "/sys/module/vfio" ]; then
        print_status $YELLOW "Warning: No UIO or VFIO modules loaded"
        print_status $YELLOW "DPDK tests may run in no-PCI mode"
    fi
}

# Function to build tests
build_tests() {
    print_status $BLUE "Building tests..."
    
    cd "$TEST_DIR/../.."
    
    # Build with meson (tests are included)
    if [ ! -d "build" ]; then
        print_status $YELLOW "Setting up meson build directory..."
        meson setup build
    fi
    
    if meson compile -C build; then
        print_status $GREEN "Tests built successfully"
    else
        print_status $RED "Failed to build tests"
        exit 1
    fi
}

# Function to run BPE tests
run_bpe_tests() {
    print_status $BLUE "Running BPE tokenizer tests..."
    
    cd "$TEST_DIR/../.."
    if sudo ./build/tests/dpdk/bpe_test --log-level=$LOG_LEVEL --no-pci; then
        print_status $GREEN "BPE tests completed successfully"
        return 0
    else
        print_status $RED "BPE tests failed"
        return 1
    fi
}

# Function to run WordPiece tests
run_wordpiece_tests() {
    print_status $BLUE "Running WordPiece tokenizer tests..."
    
    cd "$TEST_DIR/../.."
    if sudo ./build/tests/dpdk/wordpiece_test --log-level=$LOG_LEVEL --no-pci; then
        print_status $GREEN "WordPiece tests completed successfully"
        return 0
    else
        print_status $RED "WordPiece tests failed"
        return 1
    fi
}

# Function to run memory checks
run_memory_checks() {
    print_status $BLUE "Running memory checks..."
    
    cd "$TEST_DIR/../.."
    
    if command -v valgrind >/dev/null 2>&1; then
        # Run basic memory check on BPE
        print_status $YELLOW "Checking BPE memory usage..."
        print_status $YELLOW "Using AddressSanitizer for BPE memory checks..."
        ASAN_OPTIONS=detect_leaks=1 ./build/tests/dpdk/bpe_test --log-level=1 --no-pci 2>bpe_memcheck.log
        
        if [ $? -eq 0 ]; then
            print_status $GREEN "BPE memory check passed"
        else
            print_status $RED "BPE memory check failed (see bpe_memcheck.log)"
        fi
        
        # Run basic memory check on WordPiece
        print_status $YELLOW "Checking WordPiece memory usage..."
        valgrind --tool=memcheck --leak-check=summary --error-exitcode=1 \
            ./build/tests/dpdk/wordpiece_test --log-level=1 --no-pci >/dev/null 2>wordpiece_memcheck.log
        
        if [ $? -eq 0 ]; then
            print_status $GREEN "WordPiece memory check passed"
        else
            print_status $RED "WordPiece memory check failed (see wordpiece_memcheck.log)"
        fi
    else
        print_status $YELLOW "Valgrind not available, skipping memory checks"
    fi
}

# Function to generate test report
generate_report() {
    print_status $BLUE "Generating test report..."
    
    cd "$TEST_DIR/../.."
    local report_file="test_report_$(date +%Y%m%d_%H%M%S).txt"
    
    {
        echo "DPDKTokenizer Test Report"
        echo "========================="
        echo "Date: $(date)"
        echo "System: $(uname -a)"
        echo "DPDK Version: $(pkg-config --modversion libdpdk 2>/dev/null || echo 'Unknown')"
        echo ""
        echo "Test Results:"
        echo "============="
    } > "$report_file"
    
    # Run tests and capture results
    if sudo ./build/tests/dpdk/bpe_test --log-level=1 --no-pci >> "$report_file" 2>&1; then
        echo "BPE Tests: PASSED" >> "$report_file"
    else
        echo "BPE Tests: FAILED" >> "$report_file"
    fi
    
    if sudo ./build/tests/dpdk/wordpiece_test --log-level=1 --no-pci >> "$report_file" 2>&1; then
        echo "WordPiece Tests: PASSED" >> "$report_file"
    else
        echo "WordPiece Tests: FAILED" >> "$report_file"
    fi
    
    print_status $GREEN "Test report generated: $report_file"
}

# Function to show usage
show_usage() {
    echo "Usage: $0 [OPTIONS] [COMMAND]"
    echo ""
    echo "Commands:"
    echo "  build     - Build tests only"
    echo "  test      - Run all tests (default)"
    echo "  bpe       - Run BPE tests only"
    echo "  wordpiece - Run WordPiece tests only"
    echo "  memcheck  - Run memory leak checks"
    echo "  report    - Generate detailed test report"
    echo "  clean     - Clean build artifacts"
    echo "  help      - Show this help"
    echo ""
    echo "Options:"
    echo "  --log-level=N    Set DPDK log level (0-8, default: 3)"
    echo "  --huge-pages=N   Set number of huge pages (default: 512)"
    echo "  --no-root-check  Skip root privilege check"
    echo ""
    echo "Examples:"
    echo "  $0                    # Run all tests"
    echo "  $0 bpe               # Run BPE tests only"
    echo "  $0 --log-level=1     # Run with minimal logging"
    echo "  $0 memcheck          # Run with memory checking"
}

# Parse command line arguments
NO_ROOT_CHECK=false
COMMAND="test"

while [[ $# -gt 0 ]]; do
    case $1 in
        --log-level=*)
            LOG_LEVEL="${1#*=}"
            shift
            ;;
        --huge-pages=*)
            HUGE_PAGES="${1#*=}"
            shift
            ;;
        --no-root-check)
            NO_ROOT_CHECK=true
            shift
            ;;
        build|test|bpe|wordpiece|memcheck|report|clean|help)
            COMMAND=$1
            shift
            ;;
        *)
            print_status $RED "Unknown option: $1"
            show_usage
            exit 1
            ;;
    esac
done

# Check root privileges for DPDK tests
if [ "$NO_ROOT_CHECK" = false ] && [ "$EUID" -ne 0 ] && [[ "$COMMAND" =~ ^(test|bpe|wordpiece|memcheck|report)$ ]]; then
    print_status $YELLOW "DPDK tests typically require root privileges"
    print_status $YELLOW "Re-running with sudo..."
    exec sudo -E "$0" "$@" --no-root-check
fi

# Main execution
case $COMMAND in
    build)
        check_dpdk_setup
        build_tests
        ;;
    test)
        check_dpdk_setup
        build_tests
        bpe_result=0
        wordpiece_result=0
        
        run_bpe_tests || bpe_result=$?
        run_wordpiece_tests || wordpiece_result=$?
        
        if [ $bpe_result -eq 0 ] && [ $wordpiece_result -eq 0 ]; then
            print_status $GREEN "All tests PASSED!"
            exit 0
        else
            print_status $RED "Some tests FAILED!"
            exit 1
        fi
        ;;
    bpe)
        check_dpdk_setup
        build_tests
        run_bpe_tests
        ;;
    wordpiece)
        check_dpdk_setup
        build_tests
        run_wordpiece_tests
        ;;
    memcheck)
        check_dpdk_setup
        build_tests
        run_memory_checks
        ;;
    report)
        check_dpdk_setup
        build_tests
        generate_report
        ;;
    clean)
        cd "$TEST_DIR"
        make clean
        rm -f *.log test_report_*.txt
        print_status $GREEN "Cleaned build artifacts"
        ;;
    help)
        show_usage
        ;;
    *)
        print_status $RED "Unknown command: $COMMAND"
        show_usage
        exit 1
        ;;
esac
