from typing import List, Tuple, Set
import numpy as np
from multiprocessing import Process, Manager, Queue
import random
import time
from collections import defaultdict


class PPLSDC:
    """Improved Parallel Pareto Local Search with Decomposition and Cooperation"""
    
    def __init__(self, problem, **parameters):
        self.problem = problem
        self.num_processes = parameters.get('num_processes', 4)
        self.max_iterations = parameters.get('max_iterations', 200)
        self.cooperation_frequency = parameters.get('cooperation_frequency', 10)
        self.decomposition_strategy = parameters.get('decomposition', 'objective')
        
        # Problem-specific parameters
        if hasattr(problem, 'n_cities'):  # TSP
            self.is_tsp = True
            self.problem_size = problem.n_cities
        else:  # Knapsack
            self.is_tsp = False
            self.problem_size = problem.n_items
    
    def run(self):
        """Main algorithm execution with dynamic subregion adjustment"""
        manager = Manager()
        shared_archive = manager.list()
        convergence_flag = manager.Value('i', 0)
        region_info = manager.dict()  # For dynamic region adjustment
        
        # Generate diverse initial solutions
        initial_solutions = self._generate_initial_solutions()
        
        # Initialize shared archive with initial solutions
        for sol in initial_solutions:
            obj = self.problem.evaluate(sol)
            shared_archive.append((sol, obj))
        
        # Initial decomposition of objective space
        regions = self._decompose_objective_space()
        for i, region in enumerate(regions):
            region_info[i] = region
        
        # Start master process for dynamic adjustment (if enabled)
        if self.num_processes > 2:
            master_process = Process(target=self._master_process,
                                   args=(shared_archive, region_info, convergence_flag))
            master_process.start()
        
        # Start worker processes
        processes = []
        for i in range(self.num_processes):
            initial_sol = initial_solutions[i % len(initial_solutions)]
            region_idx = i % len(regions)  # Ensure valid index
            
            p = Process(target=self._run_local_search,
                       args=(i, region_idx, regions[region_idx], region_info, initial_sol,
                            shared_archive, convergence_flag))
            processes.append(p)
            p.start()
        
        # Wait for completion
        for p in processes:
            p.join()
            
        if self.num_processes > 2:
            convergence_flag.value = 1  # Signal master to stop
            master_process.join()
        
        # Convert to list and remove duplicates
        final_archive = self._remove_duplicates(list(shared_archive))
        
        # Final filtering to ensure Pareto optimality
        return self._filter_dominated(final_archive)
    
    def _master_process(self, shared_archive, region_info, convergence_flag):
        """Master process for dynamic subregion adjustment"""
        iteration = 0
        adjustment_frequency = 20  # Adjust regions every 20 iterations
        
        while convergence_flag.value == 0:
            time.sleep(1)  # Check every second
            iteration += 1
            
            if iteration % adjustment_frequency == 0:
                # Analyze current Pareto front approximation
                current_solutions = list(shared_archive)
                if len(current_solutions) > 10:
                    self._adjust_subregions(current_solutions, region_info)
    
    def _adjust_subregions(self, solutions, region_info):
        """Dynamically adjust subregions based on current solutions"""
        # Analyze distribution of solutions
        objectives = [obj for _, obj in solutions]
        
        if not objectives:
            return
        
        # Find gaps in the Pareto front
        obj1_values = sorted([obj[0] for obj in objectives])
        obj2_values = sorted([obj[1] for obj in objectives])
        
        # Identify sparse regions
        gaps = []
        for i in range(len(obj1_values) - 1):
            gap_size = obj1_values[i+1] - obj1_values[i]
            if gap_size > np.mean(np.diff(obj1_values)) * 2:
                # Large gap found
                mid_obj1 = (obj1_values[i] + obj1_values[i+1]) / 2
                # Estimate corresponding obj2
                idx = i
                mid_obj2 = (obj2_values[idx] + obj2_values[min(idx+1, len(obj2_values)-1)]) / 2
                gaps.append((mid_obj1, mid_obj2))
        
        # Adjust regions to focus on gaps
        if gaps and len(region_info) > 0:
            # Reassign some regions to explore gaps
            num_regions_to_adjust = min(len(gaps), len(region_info) // 2)
            
            for i in range(num_regions_to_adjust):
                if i < len(gaps):
                    gap = gaps[i]
                    # Calculate weight that would focus on this gap
                    if self.is_tsp:  # Minimization
                        new_weight = gap[0] / (gap[0] + gap[1]) if gap[0] + gap[1] > 0 else 0.5
                    else:  # Maximization  
                        new_weight = gap[0] / (gap[0] + gap[1]) if gap[0] + gap[1] > 0 else 0.5
                    
                    # Update region
                    region_info[i] = {'weight': np.clip(new_weight, 0.01, 0.99), 
                                     'focus': 'weighted',
                                     'adjusted': True}
    
    def _run_local_search(self, process_id, region_idx, region, region_info, initial_solution,
                         shared_archive, convergence_flag):
        """Local search for each process with region focus"""
        local_archive = [(initial_solution, self.problem.evaluate(initial_solution))]
        visited = {tuple(initial_solution)}
        
        iteration = 0
        stagnation = 0
        last_archive_size = 1
        
        while iteration < self.max_iterations:# and stagnation < 20:
            # Check global convergence
            if convergence_flag.value > 0:
                break
            
            # Get current region (may have been adjusted by master process)
            if region_idx in region_info:
                current_region = dict(region_info[region_idx])  # Copy to avoid concurrent access issues
            else:
                current_region = region
            
            # Cooperation phase
            if iteration % self.cooperation_frequency == 0 and iteration > 0:
                self._cooperate(local_archive, shared_archive, current_region)
            
            # Select solution based on region preference
            if not local_archive:
                break
                
            current_sol, current_obj = self._select_solution(local_archive, current_region)
            
            # Explore neighborhood
            new_solutions = self._explore_neighborhood(current_sol, current_obj, visited)
            
            # Filter solutions based on region
            region_solutions = self._filter_by_region(new_solutions, current_region)
            
            # Update local archive
            archive_updated = False
            for sol, obj in region_solutions:
                if self._update_archive(sol, obj, local_archive):
                    archive_updated = True
            
            # Check stagnation
            if len(local_archive) == last_archive_size:
                stagnation += 1
            else:
                stagnation = 0
                last_archive_size = len(local_archive)
            
            iteration += 1
        
        # Final sharing
        for sol, obj in local_archive:
            if not any(np.array_equal(obj, shared_obj) for _, shared_obj in shared_archive):
                shared_archive.append((sol, obj))
    
    def _decompose_objective_space(self):
        """Decompose objective space into regions"""
        if self.decomposition_strategy == 'objective':
            # Simple objective-based decomposition
            regions = []
            for i in range(self.num_processes):
                weight = i / (self.num_processes - 1) if self.num_processes > 1 else 0.5
                regions.append({'weight': weight, 'focus': 'weighted'})
        else:
            # Cone-based decomposition
            regions = []
            for i in range(self.num_processes):
                angle = i * np.pi / (2 * (self.num_processes - 1)) if self.num_processes > 1 else np.pi/4
                regions.append({'angle': angle, 'focus': 'cone'})
        
        return regions
    
    def _select_solution(self, archive, region):
        """Select solution based on region preference"""
        if not archive:
            return None, None
        
        if region['focus'] == 'weighted':
            # Select based on weighted sum
            weight = region['weight']
            best_idx = 0
            best_score = float('-inf')
            
            for i, (sol, obj) in enumerate(archive):
                if self.is_tsp:  # Minimization
                    score = -(weight * obj[0] + (1 - weight) * obj[1])
                else:  # Maximization
                    score = weight * obj[0] + (1 - weight) * obj[1]
                
                if score > best_score:
                    best_score = score
                    best_idx = i
        else:
            # Random selection with bias towards less crowded areas
            distances = self._calculate_crowding_distances(archive)
            # Tournament selection based on crowding distance
            tournament_size = min(3, len(archive))
            candidates = random.sample(range(len(archive)), tournament_size)
            best_idx = max(candidates, key=lambda i: distances[i])
        
        return archive.pop(best_idx)
    
    def _explore_neighborhood(self, solution, objectives, visited):
        """Explore neighborhood and return non-visited solutions"""
        neighbors = []
        
        if self.is_tsp:
            # 2-opt moves for TSP
            n = len(solution)
            for i in range(n - 1):
                for j in range(i + 2, min(i + 10, n)):  # Limit search radius
                    if j - i == n - 1:  # Avoid reversing entire tour
                        continue
                    
                    # Create new solution
                    new_sol = solution[:i] + solution[i:j][::-1] + solution[j:]
                    
                    if tuple(new_sol) not in visited:
                        visited.add(tuple(new_sol))
                        new_obj = self.problem.evaluate(new_sol)
                        neighbors.append((new_sol, new_obj))
        else:
            # Bit flip and swap for Knapsack
            n = len(solution)
            current_weight = sum(w * x for w, x in zip(self.problem.weights, solution))
            
            # Single bit flips
            for i in range(n):
                new_sol = solution.copy()
                new_sol[i] = 1 - new_sol[i]
                
                # Check feasibility
                if new_sol[i] == 1:
                    new_weight = current_weight + self.problem.weights[i]
                else:
                    new_weight = current_weight - self.problem.weights[i]
                
                if new_weight <= self.problem.capacity and tuple(new_sol) not in visited:
                    visited.add(tuple(new_sol))
                    new_obj = self.problem.evaluate(new_sol)
                    neighbors.append((new_sol, new_obj))
            
            # Swap moves (limit to reduce computation)
            num_swaps = min(50, n * (n - 1) // 4)
            for _ in range(num_swaps):
                i, j = random.sample(range(n), 2)
                if solution[i] != solution[j]:
                    new_sol = solution.copy()
                    new_sol[i], new_sol[j] = new_sol[j], new_sol[i]
                    
                    # Check feasibility
                    new_weight = current_weight
                    if solution[i] == 0:  # i: 0->1, j: 1->0
                        new_weight += self.problem.weights[i] - self.problem.weights[j]
                    else:  # i: 1->0, j: 0->1
                        new_weight += self.problem.weights[j] - self.problem.weights[i]
                    
                    if new_weight <= self.problem.capacity and tuple(new_sol) not in visited:
                        visited.add(tuple(new_sol))
                        new_obj = self.problem.evaluate(new_sol)
                        neighbors.append((new_sol, new_obj))
        
        return neighbors
    
    def _filter_by_region(self, solutions, region):
        """Filter solutions based on region preference"""
        if not solutions or region['focus'] != 'weighted':
            return solutions
        
        # Keep all non-dominated solutions plus some that fit the region
        weight = region['weight']
        scored_solutions = []
        
        for sol, obj in solutions:
            if self.is_tsp:  # Minimization
                score = -(weight * obj[0] + (1 - weight) * obj[1])
            else:  # Maximization
                score = weight * obj[0] + (1 - weight) * obj[1]
            scored_solutions.append((score, sol, obj))
        
        # Sort by score and keep top solutions
        scored_solutions.sort(reverse=True)
        
        # Keep top 50% plus any non-dominated solutions
        cutoff = max(len(scored_solutions) // 2, 10)
        filtered = [(sol, obj) for _, sol, obj in scored_solutions[:cutoff]]
        
        # Also keep non-dominated solutions from the rest
        for _, sol, obj in scored_solutions[cutoff:]:
            if not self._is_dominated(obj, filtered):
                filtered.append((sol, obj))
        
        return filtered
    
    def _cooperate(self, local_archive, shared_archive, region):
        """Cooperation between processes with distributed topology"""
        # Distributed topology: only share with neighboring processes
        # This is more faithful to the paper's approach
        
        # Get solutions from shared archive
        shared_solutions = list(shared_archive)
        
        # Import solutions based on region compatibility
        imported = 0
        for sol, obj in shared_solutions:
            # Check if solution fits this process's region
            if self._is_compatible_with_region(obj, region):
                if not any(np.array_equal(obj, local_obj) for _, local_obj in local_archive):
                    if len(local_archive) < 50 and imported < 5:  # Limit imports
                        local_archive.append((sol, obj))
                        imported += 1
        
        # Export best local solutions to shared archive
        if local_archive:
            # Sort by quality in this region
            if region['focus'] == 'weighted':
                weight = region['weight']
                local_sorted = sorted(local_archive, 
                                    key=lambda x: weight * x[1][0] + (1-weight) * x[1][1],
                                    reverse=not self.is_tsp)
            else:
                local_sorted = local_archive
            
            # Share top solutions
            for sol, obj in local_sorted[:3]:  # Share only top 3
                if not any(np.array_equal(obj, shared_obj) for _, shared_obj in shared_archive):
                    shared_archive.append((sol, obj))
    
    def _is_compatible_with_region(self, objectives, region):
        """Check if solution is compatible with region"""
        if region['focus'] != 'weighted':
            return True
            
        weight = region['weight']
        # Check if solution is in the "cone" of this weight vector
        # This implements a form of distributed topology
        
        if self.is_tsp:  # Minimization
            score = weight * objectives[0] + (1 - weight) * objectives[1]
            # Check neighboring weights
            for w in [weight - 0.2, weight, weight + 0.2]:
                if 0 <= w <= 1:
                    neighbor_score = w * objectives[0] + (1 - w) * objectives[1]
                    if abs(score - neighbor_score) < score * 0.1:  # Within 10%
                        return True
        else:  # Maximization
            score = weight * objectives[0] + (1 - weight) * objectives[1]
            for w in [weight - 0.2, weight, weight + 0.2]:
                if 0 <= w <= 1:
                    neighbor_score = w * objectives[0] + (1 - w) * objectives[1]
                    if abs(score - neighbor_score) < score * 0.1:
                        return True
        
        return False
    
    def _update_archive(self, solution, objectives, archive):
        """Update archive with new solution"""
        # Check if dominated
        if self._is_dominated(objectives, archive):
            return False
        
        # Remove dominated solutions
        archive[:] = [(s, o) for s, o in archive 
                     if not self._dominates(objectives, o)]
        
        # Add new solution
        archive.append((solution, objectives))
        return True
    
    def _dominates(self, obj1, obj2):
        """Check if obj1 dominates obj2"""
        if self.is_tsp:  # Minimization
            return all(o1 <= o2 for o1, o2 in zip(obj1, obj2)) and \
                   any(o1 < o2 for o1, o2 in zip(obj1, obj2))
        else:  # Maximization
            return all(o1 >= o2 for o1, o2 in zip(obj1, obj2)) and \
                   any(o1 > o2 for o1, o2 in zip(obj1, obj2))
    
    def _is_dominated(self, objectives, archive):
        """Check if objectives are dominated by any solution in archive"""
        return any(self._dominates(obj, objectives) for _, obj in archive)
    
    def _calculate_crowding_distances(self, archive):
        """Calculate crowding distances for solutions"""
        n = len(archive)
        if n <= 2:
            return [float('inf')] * n
        
        distances = [0.0] * n
        
        for obj_idx in range(2):
            # Sort by objective
            sorted_indices = sorted(range(n), key=lambda i: archive[i][1][obj_idx])
            
            # Boundary points
            distances[sorted_indices[0]] = float('inf')
            distances[sorted_indices[-1]] = float('inf')
            
            # Calculate range
            obj_range = (archive[sorted_indices[-1]][1][obj_idx] - 
                        archive[sorted_indices[0]][1][obj_idx])
            
            if obj_range > 0:
                for i in range(1, n - 1):
                    distances[sorted_indices[i]] += (
                        archive[sorted_indices[i + 1]][1][obj_idx] -
                        archive[sorted_indices[i - 1]][1][obj_idx]
                    ) / obj_range
        
        return distances
    
    def _generate_initial_solutions(self):
        """Generate diverse initial solutions"""
        solutions = []
        
        if self.is_tsp:
            # Nearest neighbor heuristic for each objective
            for obj_idx in range(2):
                for start in range(min(self.num_processes, self.problem_size)):
                    sol = self._nearest_neighbor_tsp(start, obj_idx)
                    solutions.append(sol)
            
            # Random solutions
            while len(solutions) < self.num_processes * 2:
                sol = list(range(self.problem_size))
                random.shuffle(sol)
                solutions.append(sol)
        else:
            # Greedy solutions for each objective
            for obj_idx in range(2):
                sol = self._greedy_knapsack(obj_idx)
                solutions.append(sol)
            
            # Mixed greedy
            sol = self._greedy_knapsack(2)  # Balanced
            solutions.append(sol)
            
            # Random solutions with different densities
            for density in [0.2, 0.4, 0.6, 0.8]:
                sol = self._random_knapsack(density)
                solutions.append(sol)
        
        return solutions[:self.num_processes * 2]
    
    def _nearest_neighbor_tsp(self, start, obj_idx):
        """Nearest neighbor heuristic for TSP"""
        n = self.problem_size
        unvisited = set(range(n))
        tour = [start]
        unvisited.remove(start)
        
        distances = self.problem.distances1 if obj_idx == 0 else self.problem.distances2
        
        while unvisited:
            current = tour[-1]
            nearest = min(unvisited, key=lambda x: distances[current][x])
            tour.append(nearest)
            unvisited.remove(nearest)
        
        return tour
    
    def _greedy_knapsack(self, obj_idx):
        """Greedy heuristic for knapsack"""
        n = self.problem_size
        solution = [0] * n
        
        # Calculate efficiency ratios
        ratios = []
        for i in range(n):
            if self.problem.weights[i] > 0:
                if obj_idx == 0:
                    ratio = self.problem.values[0][i] / self.problem.weights[i]
                elif obj_idx == 1:
                    ratio = self.problem.values[1][i] / self.problem.weights[i]
                else:  # Balanced
                    ratio = (self.problem.values[0][i] + self.problem.values[1][i]) / (2 * self.problem.weights[i])
                ratios.append((ratio, i))
        
        # Sort by ratio (descending)
        ratios.sort(reverse=True)
        
        # Pack items
        capacity_used = 0
        for ratio, i in ratios:
            if capacity_used + self.problem.weights[i] <= self.problem.capacity:
                solution[i] = 1
                capacity_used += self.problem.weights[i]
        
        return solution
    
    def _random_knapsack(self, density):
        """Random knapsack solution with given density"""
        n = self.problem_size
        solution = [0] * n
        
        # Randomly select items
        items = list(range(n))
        random.shuffle(items)
        
        capacity_used = 0
        target_items = int(n * density)
        
        for i in items[:target_items]:
            if capacity_used + self.problem.weights[i] <= self.problem.capacity:
                solution[i] = 1
                capacity_used += self.problem.weights[i]
        
        return solution
    
    def _remove_duplicates(self, archive):
        """Remove duplicate solutions from archive"""
        unique = []
        seen_objectives = set()
        
        for sol, obj in archive:
            obj_tuple = tuple(round(o, 6) for o in obj)
            if obj_tuple not in seen_objectives:
                seen_objectives.add(obj_tuple)
                unique.append((sol, obj))
        
        return unique
    
    def _filter_dominated(self, archive):
        """Final filtering to ensure Pareto optimality"""
        non_dominated = []
        
        for i, (sol1, obj1) in enumerate(archive):
            dominated = False
            for j, (sol2, obj2) in enumerate(archive):
                if i != j and self._dominates(obj2, obj1):
                    dominated = True
                    break
            
            if not dominated:
                non_dominated.append((sol1, obj1))
        
        return non_dominated
      

class PPLSDCforCVRP:
    """
    Enhanced PPLS/D-C specifically tuned for CVRP
    """
    
    def __init__(self, problem, **parameters):
        self.problem = problem
        
        # Core parameters
        self.num_processes = parameters.get('num_processes', 4)
        self.max_iterations = parameters.get('max_iterations', 200)
        self.cooperation_frequency = parameters.get('cooperation_frequency', 10)
        
        # CRITICAL: Increase neighborhood size for CVRP
        # CVRP needs MORE exploration than TSP due to route structure
        self.neighborhood_size = parameters.get('neighborhood_size', 300)  # Increased from 100
        
        # Problem info
        self.problem_size = problem.n_customers
        self.is_minimization = True  # CVRP minimizes both objectives
        self.n_objectives = 2
        
        # Archive management
        self.archive_size = parameters.get('archive_size', 100)
        self.patience = parameters.get('patience', 30)  # Increased patience
    
    def run(self):
        """Main execution"""
        manager = Manager()
        
        # Shared data structures
        shared_archive = manager.list()
        process_archives = manager.dict()
        subregions = manager.dict()
        convergence_flag = manager.Value('i', 0)
        
        # Generate weight vectors
        weight_vectors = self._generate_weight_vectors()
        for i in range(self.num_processes):
            subregions[i] = weight_vectors[i]
            process_archives[i] = manager.list()
        
        # Generate DIVERSE initial solutions
        initial_solutions = self._generate_diverse_initial_solutions()
        
        # Create communication topology
        topology = self._create_topology()
        
        processes = []
        
        try:
            # Start worker processes
            for i in range(self.num_processes):
                initial_sol = initial_solutions[i % len(initial_solutions)]
                neighbors = topology[i]
                
                p = Process(
                    target=self._worker_process,
                    args=(i, subregions, neighbors, initial_sol, 
                          process_archives, shared_archive, convergence_flag)
                )
                processes.append(p)
                p.start()
            
            # Wait for workers
            for p in processes:
                p.join()
                
        except Exception as e:
            print(f"Error: {e}")
            convergence_flag.value = 1
            for p in processes:
                if p.is_alive():
                    p.terminate()
        
        # Extract final Pareto front
        return self._extract_pareto_front(list(shared_archive))
    
    def _worker_process(self, process_id, subregions, neighbors, initial_solution,
                       process_archives, shared_archive, convergence_flag):
        """Enhanced worker process with better CVRP operators"""
        local_archive = [initial_solution]
        current_solution = initial_solution
        
        iteration = 0
        stagnation = 0
        best_value = float('inf')
        
        while iteration < self.max_iterations and stagnation < self.patience:
            if convergence_flag.value > 0:
                break
            
            # Get weight vector
            weight_vector = subregions[process_id]
            
            # Generate ENHANCED neighborhood for CVRP
            neighborhood = self._generate_enhanced_cvrp_neighborhood(current_solution)
            
            # Find best neighbor
            improved = False
            best_neighbor = None
            best_neighbor_value = best_value
            
            for neighbor in neighborhood:
                obj = self.problem.evaluate(neighbor)
                if obj[0] == float('inf'):  # Invalid solution
                    continue
                
                value = self._scalarize(obj, weight_vector)
                
                if value < best_neighbor_value:
                    best_neighbor = neighbor
                    best_neighbor_value = value
                    improved = True
            
            if improved:
                current_solution = best_neighbor
                best_value = best_neighbor_value
                self._update_archive(current_solution, local_archive)
                stagnation = 0
            else:
                stagnation += 1
            
            # Cooperation phase
            if iteration % self.cooperation_frequency == 0 and iteration > 0:
                # Share solutions
                self._share_solutions(process_id, local_archive, process_archives, 
                                     weight_vector)
                
                # Import from neighbors
                for neighbor_id in neighbors:
                    if neighbor_id in process_archives:
                        try:
                            neighbor_archive = list(process_archives[neighbor_id])
                            for sol in neighbor_archive[:5]:
                                self._update_archive(sol, local_archive)
                        except:
                            continue
                
                # Select new current solution from archive
                if local_archive:
                    current_solution = self._select_by_weight(local_archive, weight_vector)
                    best_value = self._scalarize(self.problem.evaluate(current_solution), 
                                                 weight_vector)
            
            # Periodically update shared archive
            if iteration % (self.cooperation_frequency * 2) == 0:
                for sol in local_archive[:10]:
                    obj = self.problem.evaluate(sol)
                    if obj[0] != float('inf'):
                        shared_archive.append((sol, obj))
            
            iteration += 1
        
        # Final update
        for sol in local_archive:
            obj = self.problem.evaluate(sol)
            if obj[0] != float('inf'):
                shared_archive.append((sol, obj))
    
    def _generate_enhanced_cvrp_neighborhood(self, solution):
        """
        ENHANCED neighborhood generation with proper CVRP operators
        Matches the diversity of TSP 2-opt exploration
        """
        neighbors = []
        routes = solution
        n_routes = len(routes)
        
        # 1. INTRA-ROUTE 2-OPT (more comprehensive)
        for route_idx, route in enumerate(routes):
            if len(route) < 2:
                continue
            
            route_len = len(route)
            # Try MORE 2-opt moves per route
            max_moves = min(route_len * (route_len - 1) // 2, 50)
            
            for _ in range(max_moves):
                if len(neighbors) >= self.neighborhood_size // 3:
                    break
                
                i = random.randint(0, route_len - 2)
                j = random.randint(i + 1, route_len)
                
                new_routes = [r.copy() for r in routes]
                new_routes[route_idx] = route[:i] + route[i:j][::-1] + route[j:]
                neighbors.append(new_routes)
        
        # 2. RELOCATE (move single customer)
        for _ in range(self.neighborhood_size // 3):
            if n_routes < 2:
                break
            
            from_idx = random.randint(0, n_routes - 1)
            if not routes[from_idx]:
                continue
            
            to_idx = random.randint(0, n_routes - 1)
            from_route = routes[from_idx]
            to_route = routes[to_idx] if to_idx < len(routes) else []
            
            if not from_route:
                continue
            
            cust_idx = random.randint(0, len(from_route) - 1)
            customer = from_route[cust_idx]
            
            # Check capacity
            to_demand = sum(self.problem.customers[c].demand for c in to_route)
            if to_demand + self.problem.customers[customer].demand <= self.problem.vehicle_capacity:
                # Try multiple insert positions
                for insert_pos in range(len(to_route) + 1):
                    new_routes = [r.copy() for r in routes]
                    new_routes[from_idx] = from_route[:cust_idx] + from_route[cust_idx+1:]
                    new_routes[to_idx] = to_route[:insert_pos] + [customer] + to_route[insert_pos:]
                    
                    # Remove empty routes
                    new_routes = [r for r in new_routes if r]
                    neighbors.append(new_routes)
                    
                    if len(neighbors) >= 2 * self.neighborhood_size // 3:
                        break
        
        # 3. OR-OPT (relocate sequences of 2-3 customers)
        for seq_len in [2, 3]:
            for _ in range(20):  # Try 20 or-opt moves
                if n_routes < 1:
                    break
                
                from_idx = random.randint(0, n_routes - 1)
                from_route = routes[from_idx]
                
                if len(from_route) < seq_len:
                    continue
                
                seq_start = random.randint(0, len(from_route) - seq_len)
                sequence = from_route[seq_start:seq_start + seq_len]
                seq_demand = sum(self.problem.customers[c].demand for c in sequence)
                
                # Try to insert into another route
                to_idx = random.randint(0, n_routes - 1)
                to_route = routes[to_idx] if to_idx < len(routes) else []
                
                to_demand = sum(self.problem.customers[c].demand for c in to_route)
                if from_idx == to_idx:
                    # Insert elsewhere in same route
                    for insert_pos in range(len(from_route) - seq_len + 1):
                        if insert_pos != seq_start:
                            new_route = (from_route[:seq_start] + 
                                       from_route[seq_start + seq_len:])
                            new_route = (new_route[:insert_pos] + sequence + 
                                       new_route[insert_pos:])
                            new_routes = [r.copy() for r in routes]
                            new_routes[from_idx] = new_route
                            neighbors.append(new_routes)
                elif to_demand + seq_demand <= self.problem.vehicle_capacity:
                    insert_pos = random.randint(0, len(to_route))
                    
                    new_routes = [r.copy() for r in routes]
                    new_routes[from_idx] = (from_route[:seq_start] + 
                                           from_route[seq_start + seq_len:])
                    new_routes[to_idx] = (to_route[:insert_pos] + sequence + 
                                         to_route[insert_pos:])
                    
                    # Remove empty routes
                    new_routes = [r for r in new_routes if r]
                    neighbors.append(new_routes)
        
        # 4. SWAP (exchange customers between routes)
        for _ in range(self.neighborhood_size // 6):
            if n_routes < 2:
                break
            
            route1_idx = random.randint(0, n_routes - 1)
            route2_idx = random.randint(0, n_routes - 1)
            
            if route1_idx == route2_idx:
                continue
            
            route1 = routes[route1_idx]
            route2 = routes[route2_idx]
            
            if not route1 or not route2:
                continue
            
            cust1_idx = random.randint(0, len(route1) - 1)
            cust2_idx = random.randint(0, len(route2) - 1)
            
            cust1 = route1[cust1_idx]
            cust2 = route2[cust2_idx]
            
            # Check capacity
            route1_demand = sum(self.problem.customers[c].demand for c in route1)
            route2_demand = sum(self.problem.customers[c].demand for c in route2)
            
            new_route1_demand = (route1_demand - self.problem.customers[cust1].demand + 
                                self.problem.customers[cust2].demand)
            new_route2_demand = (route2_demand - self.problem.customers[cust2].demand + 
                                self.problem.customers[cust1].demand)
            
            if (new_route1_demand <= self.problem.vehicle_capacity and 
                new_route2_demand <= self.problem.vehicle_capacity):
                
                new_routes = [r.copy() for r in routes]
                new_routes[route1_idx][cust1_idx] = cust2
                new_routes[route2_idx][cust2_idx] = cust1
                neighbors.append(new_routes)
        
        # 5. CROSS EXCHANGE (swap sequences between routes)
        for _ in range(20):
            if n_routes < 2:
                break
            
            route1_idx = random.randint(0, n_routes - 1)
            route2_idx = random.randint(0, n_routes - 1)
            
            if route1_idx == route2_idx:
                continue
            
            route1 = routes[route1_idx]
            route2 = routes[route2_idx]
            
            if len(route1) < 2 or len(route2) < 2:
                continue
            
            # Random segments
            seg1_len = random.randint(1, min(3, len(route1)))
            seg2_len = random.randint(1, min(3, len(route2)))
            
            seg1_start = random.randint(0, len(route1) - seg1_len)
            seg2_start = random.randint(0, len(route2) - seg2_len)
            
            seg1 = route1[seg1_start:seg1_start + seg1_len]
            seg2 = route2[seg2_start:seg2_start + seg2_len]
            
            # Check capacity
            seg1_demand = sum(self.problem.customers[c].demand for c in seg1)
            seg2_demand = sum(self.problem.customers[c].demand for c in seg2)
            
            route1_demand = sum(self.problem.customers[c].demand for c in route1)
            route2_demand = sum(self.problem.customers[c].demand for c in route2)
            
            new_route1_demand = route1_demand - seg1_demand + seg2_demand
            new_route2_demand = route2_demand - seg2_demand + seg1_demand
            
            if (new_route1_demand <= self.problem.vehicle_capacity and 
                new_route2_demand <= self.problem.vehicle_capacity):
                
                new_route1 = (route1[:seg1_start] + seg2 + 
                             route1[seg1_start + seg1_len:])
                new_route2 = (route2[:seg2_start] + seg1 + 
                             route2[seg2_start + seg2_len:])
                
                new_routes = [r.copy() for r in routes]
                new_routes[route1_idx] = new_route1
                new_routes[route2_idx] = new_route2
                neighbors.append(new_routes)
        
        return neighbors[:self.neighborhood_size]
    
    def _generate_diverse_initial_solutions(self):
        """Generate DIVERSE initial solutions using multiple heuristics"""
        solutions = []
        
        # 1. Clarke-Wright Savings (complete implementation)
        for _ in range(2):
            sol = self._clarke_wright_savings()
            solutions.append(sol)
        
        # 2. Nearest Neighbor from different starts
        for start in range(min(self.num_processes, 5)):
            sol = self._nearest_neighbor_cvrp(start + 1)
            solutions.append(sol)
        
        # 3. Sweep algorithm
        for angle_offset in [0, 45, 90, 135]:
            sol = self._sweep_heuristic(angle_offset)
            solutions.append(sol)
        
        # 4. Random but valid solutions
        for _ in range(self.num_processes):
            sol = self.problem.random_solution()
            solutions.append(sol)
        
        return solutions[:self.num_processes * 2]
    
    def _clarke_wright_savings(self):
        """Complete Clarke-Wright Savings algorithm"""
        # Start with single-customer routes
        routes = [[i] for i in range(1, self.problem.n_customers + 1)]
        
        # Calculate all savings
        savings = []
        for i in range(1, self.problem.n_customers + 1):
            for j in range(i + 1, self.problem.n_customers + 1):
                saving = (self.problem.distances[0][i] + 
                         self.problem.distances[0][j] - 
                         self.problem.distances[i][j])
                savings.append((saving, i, j))
        
        # Sort by savings (descending)
        savings.sort(reverse=True)
        
        # Merge routes based on savings
        for saving, i, j in savings:
            # Find routes containing i and j
            route_i = None
            route_j = None
            route_i_idx = None
            route_j_idx = None
            
            for idx, route in enumerate(routes):
                if i in route:
                    route_i = route
                    route_i_idx = idx
                if j in route:
                    route_j = route
                    route_j_idx = idx
            
            if route_i and route_j and route_i != route_j:
                # Check if i and j are at route ends
                i_at_end = (route_i[0] == i or route_i[-1] == i)
                j_at_end = (route_j[0] == j or route_j[-1] == j)
                
                if i_at_end and j_at_end:
                    # Check capacity
                    total_demand = (sum(self.problem.customers[c].demand for c in route_i) +
                                   sum(self.problem.customers[c].demand for c in route_j))
                    
                    if total_demand <= self.problem.vehicle_capacity:
                        # Merge routes
                        if route_i[-1] == i and route_j[0] == j:
                            new_route = route_i + route_j
                        elif route_i[0] == i and route_j[-1] == j:
                            new_route = route_j + route_i
                        elif route_i[-1] == i and route_j[-1] == j:
                            new_route = route_i + route_j[::-1]
                        elif route_i[0] == i and route_j[0] == j:
                            new_route = route_i[::-1] + route_j
                        else:
                            continue
                        
                        routes.remove(route_i)
                        routes.remove(route_j)
                        routes.append(new_route)
        
        return routes
    
    def _nearest_neighbor_cvrp(self, start_customer):
        """Nearest neighbor heuristic"""
        unassigned = set(range(1, self.problem.n_customers + 1))
        routes = []
        
        while unassigned:
            route = []
            route_demand = 0
            current_pos = 0  # Depot
            
            # Start with specific customer if still unassigned
            if start_customer in unassigned:
                first_cust = start_customer
            else:
                first_cust = min(unassigned, 
                               key=lambda c: self.problem.distances[0][c])
            
            route.append(first_cust)
            route_demand += self.problem.customers[first_cust].demand
            unassigned.remove(first_cust)
            current_pos = first_cust
            
            # Build rest of route
            while unassigned:
                # Find nearest feasible customer
                best_customer = None
                best_dist = float('inf')
                
                for customer in unassigned:
                    if (route_demand + self.problem.customers[customer].demand <= 
                        self.problem.vehicle_capacity):
                        dist = self.problem.distances[current_pos][customer]
                        if dist < best_dist:
                            best_dist = dist
                            best_customer = customer
                
                if best_customer is None:
                    break
                
                route.append(best_customer)
                route_demand += self.problem.customers[best_customer].demand
                unassigned.remove(best_customer)
                current_pos = best_customer
            
            routes.append(route)
        
        return routes
    
    def _sweep_heuristic(self, angle_offset=0):
        """Sweep algorithm with angle offset for diversity"""
        # Calculate angles from depot
        customers_with_angles = []
        for i in range(1, self.problem.n_customers + 1):
            cust = self.problem.customers[i]
            angle = np.arctan2(cust.y - 0.5, cust.x - 0.5)
            angle = (angle + angle_offset * np.pi / 180) % (2 * np.pi)
            customers_with_angles.append((angle, i))
        
        # Sort by angle
        customers_with_angles.sort()
        
        # Assign to routes
        routes = []
        current_route = []
        current_demand = 0
        
        for angle, customer in customers_with_angles:
            demand = self.problem.customers[customer].demand
            
            if current_demand + demand <= self.problem.vehicle_capacity:
                current_route.append(customer)
                current_demand += demand
            else:
                if current_route:
                    routes.append(current_route)
                current_route = [customer]
                current_demand = demand
        
        if current_route:
            routes.append(current_route)
        
        return routes
    
    # Helper methods (unchanged from original)
    def _generate_weight_vectors(self):
        """Generate weight vectors for bi-objective"""
        weights = []
        for i in range(self.num_processes):
            w1 = i / (self.num_processes - 1) if self.num_processes > 1 else 0.5
            weights.append((w1, 1 - w1))
        return weights
    
    def _create_topology(self):
        """Create ring topology"""
        topology = {}
        for i in range(self.num_processes):
            topology[i] = [(i-1) % self.num_processes, 
                          (i+1) % self.num_processes]
        return topology
    
    def _scalarize(self, objectives, weight_vector):
        """Weighted sum scalarization"""
        return sum(w * obj for w, obj in zip(weight_vector, objectives))
    
    def _share_solutions(self, process_id, local_archive, process_archives, weight_vector):
        """Share top solutions"""
        if not local_archive:
            return
        
        sorted_archive = sorted(
            local_archive,
            key=lambda sol: self._scalarize(self.problem.evaluate(sol), weight_vector)
        )
        
        process_archives[process_id] = sorted_archive[:10]
    
    def _select_by_weight(self, archive, weight_vector):
        """Select best solution by weight"""
        if not archive:
            return None
        
        return min(
            archive,
            key=lambda sol: self._scalarize(self.problem.evaluate(sol), weight_vector)
        )
    
    def _update_archive(self, solution, archive):
        """Update archive with non-dominated solutions"""
        obj = self.problem.evaluate(solution)
        
        if obj[0] == float('inf'):
            return
        
        # Remove dominated
        archive[:] = [sol for sol in archive 
                     if not self._dominates(obj, self.problem.evaluate(sol))]
        
        # Add if not dominated
        is_dominated = any(
            self._dominates(self.problem.evaluate(sol), obj) 
            for sol in archive
        )
        
        if not is_dominated:
            archive.append(solution)
            
            if len(archive) > self.archive_size:
                self._reduce_by_crowding(archive)
    
    def _dominates(self, obj1, obj2):
        """Pareto dominance for minimization"""
        return (all(o1 <= o2 for o1, o2 in zip(obj1, obj2)) and 
               any(o1 < o2 for o1, o2 in zip(obj1, obj2)))
    
    def _reduce_by_crowding(self, archive):
        """Remove most crowded solution"""
        objectives = [self.problem.evaluate(sol) for sol in archive]
        distances = self._crowding_distances(objectives)
        min_idx = np.argmin(distances)
        del archive[min_idx]
    
    def _crowding_distances(self, objectives):
        """Calculate crowding distances"""
        n = len(objectives)
        if n <= 2:
            return [float('inf')] * n
        
        distances = [0.0] * n
        
        for obj_idx in range(2):
            sorted_indices = sorted(range(n), key=lambda i: objectives[i][obj_idx])
            
            distances[sorted_indices[0]] = float('inf')
            distances[sorted_indices[-1]] = float('inf')
            
            obj_range = (objectives[sorted_indices[-1]][obj_idx] - 
                        objectives[sorted_indices[0]][obj_idx])
            
            if obj_range > 0:
                for i in range(1, n - 1):
                    distances[sorted_indices[i]] += (
                        objectives[sorted_indices[i + 1]][obj_idx] -
                        objectives[sorted_indices[i - 1]][obj_idx]
                    ) / obj_range
        
        return distances
    
    def _extract_pareto_front(self, archive):
        """Extract final Pareto front"""
        pareto_front = []
        
        for sol, obj in archive:
            if obj[0] == float('inf'):
                continue
            
            is_dominated = False
            for _, other_obj in archive:
                if (other_obj[0] != float('inf') and 
                    self._dominates(other_obj, obj) and 
                    tuple(other_obj) != tuple(obj)):
                    is_dominated = True
                    break
            
            if not is_dominated:
                pareto_front.append((sol, obj))
        
        # Remove duplicates
        unique_pf = []
        seen = set()
        for sol, obj in pareto_front:
            obj_tuple = tuple(obj)
            if obj_tuple not in seen:
                seen.add(obj_tuple)
                unique_pf.append((sol, obj))
        
        return unique_pf


def test_pplsdc(num_runs=3):
    """Test PPLS/D-C with multiple runs on different instances"""
    from MOCO.problems import BiObjectiveTSP, MultiObjectiveKnapsack
    from MOCO.evaluation import MOCOEvaluator
    
    # Updated parameters for improved implementation
    algorithm_params = {
        'num_processes': 4,          # Reduced for better efficiency
        'max_iterations': 100,       # Reduced for faster runtime
        'cooperation_frequency': 10, # How often processes share solutions
        'decomposition': 'objective' # Decomposition strategy
    }
    
    print("\nInitial Single Run Results:")
    print("===========================")
    
    # Test on TSP
    print("\nTesting PPLS/D-C on Bi-Objective TSP:")
    bitsp = BiObjectiveTSP(n_cities=20)
    pplsdc_tsp = PPLSDC(bitsp, **algorithm_params)
    
    start_time = time.time()
    solutions_tsp = pplsdc_tsp.run()
    runtime = time.time() - start_time
    
    print(f"Found {len(solutions_tsp)} non-dominated solutions in {runtime:.2f} seconds")
    # Show first 5 solutions
    for i, (_, objectives) in enumerate(solutions_tsp[:5]):
        print(f"  Solution {i+1}: ({objectives[0]:.0f}, {objectives[1]:.0f})")
    if len(solutions_tsp) > 5:
        print(f"  ... and {len(solutions_tsp) - 5} more solutions")
    
    # Test on Knapsack
    print("\nTesting PPLS/D-C on Multi-Objective Knapsack:")
    mokp = MultiObjectiveKnapsack(n_items=50, n_objectives=2, capacity=12.5)
    pplsdc_kp = PPLSDC(mokp, **algorithm_params)
    
    start_time = time.time()
    solutions_kp = pplsdc_kp.run()
    runtime = time.time() - start_time
    
    print(f"Found {len(solutions_kp)} non-dominated solutions in {runtime:.2f} seconds")
    # Show first 5 solutions
    for i, (_, objectives) in enumerate(solutions_kp[:5]):
        print(f"  Solution {i+1}: ({objectives[0]:.2f}, {objectives[1]:.2f})")
    if len(solutions_kp) > 5:
        print(f"  ... and {len(solutions_kp) - 5} more solutions")
    
    print("\n" + "="*60)
    print("Multiple Runs Evaluation:")
    print("="*60)
    
    # Test on different problem sizes
    test_configs = [
        # Problem sizes that match your benchmark
        {
            'problem_class': BiObjectiveTSP,
            'problem_name': 'BiTSP',
            'problem_params': {'n_cities': 20},  # Small
            'algorithm_params': algorithm_params
        },
        {
            'problem_class': BiObjectiveTSP,
            'problem_name': 'BiTSP',
            'problem_params': {'n_cities': 50},  # Medium
            'algorithm_params': algorithm_params
        },
        {
            'problem_class': MultiObjectiveKnapsack,
            'problem_name': 'BiKP',
            'problem_params': {'n_items': 50, 'n_objectives': 2, 'capacity': 12.5},  # Small
            'algorithm_params': algorithm_params
        },
        {
            'problem_class': MultiObjectiveKnapsack,
            'problem_name': 'BiKP',
            'problem_params': {'n_items': 100, 'n_objectives': 2, 'capacity': 25},  # Medium
            'algorithm_params': algorithm_params
        }
    ]
    
    # Initialize evaluator
    evaluator = MOCOEvaluator(reference_point=(1.2, 1.2), confidence_level=0.95)
    
    print(f"\nRunning each configuration {num_runs} times...")
    
    for config in test_configs:
        print(f"\nEvaluating PPLS/D-C on {config['problem_name']} "
              f"(size={list(config['problem_params'].values())[0]})...")
        
        result = evaluator.evaluate_algorithm(
            algorithm_class=PPLSDC,
            problem_class=config['problem_class'],
            algorithm_name="PPLS-D-C",
            parameters=config['algorithm_params'],
            problem_params=config['problem_params'],
            num_runs=num_runs
        )
    
    # Generate comprehensive report
    evaluator.generate_report()
    
    print("\n" + "="*60)
    print("Aggregated Results:")
    print("="*60)
    
    for result in evaluator.results:
        print(f"\n{result.algorithm_name} on {result.problem_name}:")
        print(f"  Problem size: {result.parameters['problem']}")
        print(f"  Average Runtime: {result.runtime:.2f} seconds")
        print(f"  Hypervolume: {result.hypervolume:.4f}")
        print(f"  Non-dominated solutions: {result.num_nondominated}")
        print(f"  Convergence: {result.metrics_over_time}")
    
    # Reliability analysis
    reliability = evaluator.generate_reliability_scores()
    
    print("\n" + "="*60)
    print("Reliability Analysis:")
    print("="*60)
    
    for key, scores in reliability.items():
        print(f"\n{key}:")
        for metric, score in scores.items():
            category = "Excellent" if score > 90 else "Good" if score > 75 else "Fair" if score > 50 else "Poor"
            print(f"  {metric}: {score:.1f}/100 ({category})")
    
    # Generate visualizations
    print("\nGenerating visualizations...")
    
    try:
        # Performance profiles
        evaluator.plot_performance_profiles(metric='hypervolume')
        
        # Pareto fronts
        evaluator.plot_pareto_front(show_all=True)
        
        # Comparison plots
        evaluator.plot_comparison()
    except Exception as e:
        print(f"Warning: Could not generate plots: {e}")
    
    return evaluator

def test_pplsdc_cvpr(num_runs=3):
    """Test PPLS/D-C with multiple runs on different instances"""
    from MOCO.problems import BiObjectiveTSP, MultiObjectiveKnapsack, BiObjectiveCVRP
    from MOCO.evaluation import MOCOEvaluator
    
    # Updated parameters for improved implementation
    algorithm_params = {
        'num_processes': 4,          # Reduced for better efficiency
        'max_iterations': 200,       # Reduced for faster runtime
        'cooperation_frequency': 10, # How often processes share solutions
        'decomposition': 'objective' # Decomposition strategy
    }
    
    print("\nInitial Single Run Results:")
    print("===========================")
    
    
    # Test on CVRP
    print("\nTesting PPLS/D-C on Bi-Objective CVRP:")
    cvrp = BiObjectiveCVRP(n_customers=20)  # Will auto-set n_vehicles
    pplsdc_cvrp = PPLSDCforCVRP(cvrp, **algorithm_params)
    
    start_time = time.time()
    solutions_cvrp = pplsdc_cvrp.run()
    runtime = time.time() - start_time
    
    print(f"Found {len(solutions_cvrp)} non-dominated solutions in {runtime:.2f} seconds")
    # Show first 5 solutions
    for i, (routes, objectives) in enumerate(solutions_cvrp[:5]):
        print(f"  Solution {i+1}: Total Distance={objectives[0]:.2f}, Makespan={objectives[1]:.2f}")
        print(f"    Routes: {routes}")
    if len(solutions_cvrp) > 5:
        print(f"  ... and {len(solutions_cvrp) - 5} more solutions")
    
    print("\n" + "="*60)
    print("Multiple Runs Evaluation:")
    print("="*60)
    
    # Test on different problem sizes
    test_configs = [
        # TSP Problems
        # {
        #     'problem_class': BiObjectiveTSP,
        #     'algorithm_class': PPLSDC,
        #     'problem_name': 'BiTSP',
        #     'problem_params': {'n_cities': 20},  # Small
        #     'algorithm_params': algorithm_params
        # },
        # {
        #     'problem_class': BiObjectiveTSP,
        #     'algorithm_class': PPLSDC,
        #     'problem_name': 'BiTSP',
        #     'problem_params': {'n_cities': 50},  # Medium
        #     'algorithm_params': algorithm_params
        # },
        # Knapsack Problems
        # {
        #     'problem_class': MultiObjectiveKnapsack,
        #     'algorithm_class': PPLSDC,
        #     'problem_name': 'BiKP',
        #     'problem_params': {'n_items': 50, 'n_objectives': 2, 'capacity': 12.5},  # Small
        #     'algorithm_params': algorithm_params
        # },
        # {
        #     'problem_class': MultiObjectiveKnapsack,
        #     'algorithm_class': PPLSDC,
        #     'problem_name': 'BiKP',
        #     'problem_params': {'n_items': 100, 'n_objectives': 2, 'capacity': 25},  # Medium
        #     'algorithm_params': algorithm_params
        # },
        # CVRP Problems
        {
            'problem_class': BiObjectiveCVRP,
            'algorithm_class': PPLSDCforCVRP,  # Use wrapper for CVRP
            'problem_name': 'BiCVRP',
            'problem_params': {'n_customers': 20},  # Small (D=30 auto-set)
            'algorithm_params': algorithm_params
        },
        # {
        #     'problem_class': BiObjectiveCVRP,
        #     'algorithm_class': PPLSDCforCVRP,  # Use wrapper for CVRP
        #     'problem_name': 'BiCVRP',
        #     'problem_params': {'n_customers': 50},  # Medium (D=40 auto-set)
        #     'algorithm_params': algorithm_params
        # }
    ]
    
    # Initialize evaluator with appropriate reference points
    # For CVRP, we need larger reference points since distances can be larger
    evaluator = MOCOEvaluator(reference_point=(40, 3), confidence_level=0.95)
    
    print(f"\nRunning each configuration {num_runs} times...")
    
    for config in test_configs:
        print(f"\nEvaluating PPLS/D-C on {config['problem_name']} "
              f"(size={list(config['problem_params'].values())[0]})...")
        
        # Use the specific algorithm class for each problem type
        result = evaluator.evaluate_algorithm(
            algorithm_class=config['algorithm_class'],
            problem_class=config['problem_class'],
            algorithm_name=f"PPLS-D-C-{config['problem_name']}",
            parameters=config['algorithm_params'],
            problem_params=config['problem_params'],
            num_runs=num_runs
        )
    
    # Generate comprehensive report
    evaluator.generate_report()
    
    print("\n" + "="*60)
    print("Aggregated Results:")
    print("="*60)
    
    for result in evaluator.results:
        print(f"\n{result.algorithm_name} on {result.problem_name}:")
        print(f"  Problem size: {result.problem_size}")
        print(f"  Average Runtime: {result.runtime:.2f} seconds")
        print(f"  Hypervolume: {result.hypervolume:.4f}")
        print(f"  Non-dominated solutions: {result.num_nondominated}")
        
        # Show statistics if available
        if result.statistics:
            if 'hypervolume' in result.statistics:
                hv_stats = result.statistics['hypervolume']
                print(f"  HV Mean±Std: {hv_stats['mean']:.4f}±{hv_stats['std_dev']:.4f}")
    
    # Reliability analysis
    reliability = evaluator.generate_reliability_scores()
    
    print("\n" + "="*60)
    print("Reliability Analysis:")
    print("="*60)
    
    for key, scores in reliability.items():
        print(f"\n{key}:")
        for metric, score in scores.items():
            category = "Excellent" if score > 90 else "Good" if score > 75 else "Fair" if score > 50 else "Poor"
            print(f"  {metric}: {score:.1f}/100 ({category})")
    
    # Generate visualizations
    print("\nGenerating visualizations...")
    
    try:
        # Performance profiles
        evaluator.plot_performance_profiles(metric='hypervolume', save_path='pplsdc_hypervolume.png')
        
        # Multi-metric profiles
        evaluator.plot_multi_metric_profiles(save_path='pplsdc_metrics.png')
        
        # Pareto fronts for each problem type
        for i, result in enumerate(evaluator.results):
            if 'TSP' in result.problem_name:
                evaluator.plot_pareto_front(result_index=i, show_all=False)
                break
    except Exception as e:
        print(f"Warning: Could not generate plots: {e}")
    
    return evaluator


def quick_test():
    """Quick test with single run for debugging"""
    print("Quick test of PPLS/D-C...")
    from MOCO.problems import BiObjectiveTSP, MultiObjectiveKnapsack
    from MOCO.evaluation import MOCOEvaluator

    # Small TSP instance
    problem = BiObjectiveTSP(n_cities=20)
    algorithm = PPLSDC(problem, num_processes=4, max_iterations=50)
    
    start = time.time()
    solutions = algorithm.run()
    runtime = time.time() - start
    
    print(f"\nResults:")
    print(f"  Runtime: {runtime:.2f} seconds")
    print(f"  Solutions found: {len(solutions)}")
    
    if solutions:
        objectives = [obj for _, obj in solutions]
        obj1_values = [obj[0] for obj in objectives]
        obj2_values = [obj[1] for obj in objectives]
        
        print(f"  Objective 1 range: [{min(obj1_values):.0f}, {max(obj1_values):.0f}]")
        print(f"  Objective 2 range: [{min(obj2_values):.0f}, {max(obj2_values):.0f}]")


def test_quick_cvpr():
    from MOCO.problems import BiObjectiveCVRP

    # Create problem
    cvrp = BiObjectiveCVRP(n_customers=20)  # Will use D=30 automatically

    # Run algorithm
    algorithm = PPLSDCforCVRP(cvrp, 
                            num_processes=4,
                            max_iterations=50,
                            cooperation_frequency=10)

    solutions = algorithm.run()
    print(f"Found {len(solutions)} non-dominated solutions")

    # Check first solution
    if solutions:
        routes, (total_dist, makespan) = solutions[0]
        print(f"Example solution: Total distance={total_dist:.2f}, Makespan={makespan:.2f}")
        print(f"Routes: {routes}")


if __name__ == "__main__":
    import time
    
    # First do a quick test
    # quick_test()
    
    # Then run full evaluation
    print("\n" + "="*60)
    print("Starting full evaluation...")
    print("="*60)
    
    # Use small number of runs for testing
    # test_pplsdc(num_runs=1)

    test_pplsdc_cvpr(num_runs=1)