"""CVRP data augmentation distributions for initialization."""

from __future__ import annotations

import numpy as np


# Geometry分布族权重配置
GEOMETRY_MIX_WEIGHTS = {
    "uniform_square": 0.30,
    "clustered_gaussian": 0.25,
    "two_scale_mixture": 0.15,
    "corridor": 0.15,
    "annulus_ring": 0.15,
}

# Demand分布族权重配置
DEMAND_MIX_WEIGHTS = {
    "uniform_small": 0.40,
    "bimodal": 0.30,
    "heavy_tail": 0.20,
    "geo_correlated": 0.10,
}

# Capacity regime权重配置
CAPACITY_REGIME_WEIGHTS = {
    "loose": 0.33,    # ρ=1.2, m_target≈n/6
    "medium": 0.34,   # ρ=1.0, m_target≈n/7
    "tight": 0.33,    # ρ=0.85, m_target≈n/8
}


def get_da_generator_code() -> str:
    """
    generate CVRP generator code using data augmentation distributions.
    
    each time generate_instances is called:
    1. randomly select a geometry distribution family to generate coordinates
    2. randomly select a demand distribution family to generate demands
    3. randomly select a capacity regime to set capacity
    4. randomly apply isometric transformation (rotate/reflect/translate/scale)
    
    Returns:
        complete generator code string
    """
    return (
        "import numpy as np\n"
        "\n"
        "def generate_instances(seeds, n_customers, capacity=None):\n"
        "    instances = []\n"
        "    \n"
        "    geometry_names = ['uniform_square', 'clustered_gaussian', 'two_scale_mixture', 'corridor', 'annulus_ring']\n"
        "    geometry_weights = [0.30, 0.25, 0.15, 0.15, 0.15]\n"
        "    \n"
        "    demand_names = ['uniform_small', 'bimodal', 'heavy_tail', 'geo_correlated']\n"
        "    demand_weights = [0.40, 0.30, 0.20, 0.10]\n"
        "    \n"
        "    regime_names = ['loose', 'medium', 'tight']\n"
        "    regime_weights = [0.33, 0.34, 0.33]\n"
        "    \n"
        "    for seed in seeds:\n"
        "        rng = np.random.default_rng(seed)\n"
        "        \n"
        "        geom_type = np.random.choice(geometry_names, p=geometry_weights)\n"
        "        if geom_type == 'uniform_square':\n"
        "            coords, depot = generate_uniform_square(n_customers, rng)\n"
        "        elif geom_type == 'clustered_gaussian':\n"
        "            coords, depot = generate_clustered_gaussian(n_customers, rng)\n"
        "        elif geom_type == 'two_scale_mixture':\n"
        "            coords, depot = generate_two_scale_mixture(n_customers, rng)\n"
        "        elif geom_type == 'corridor':\n"
        "            coords, depot = generate_corridor(n_customers, rng)\n"
        "        elif geom_type == 'annulus_ring':\n"
        "            coords, depot = generate_annulus_ring(n_customers, rng)\n"
        "        else:\n"
        "            coords, depot = generate_uniform_square(n_customers, rng)\n"
        "        \n"
        "        coords, depot = apply_isometric_transform(coords, depot, rng)\n"
        "        \n"
        "        demand_type = np.random.choice(demand_names, p=demand_weights)\n"
        "        if demand_type == 'uniform_small':\n"
        "            demands = generate_uniform_small_demands(n_customers, rng)\n"
        "        elif demand_type == 'bimodal':\n"
        "            demands = generate_bimodal_demands(n_customers, rng)\n"
        "        elif demand_type == 'heavy_tail':\n"
        "            demands = generate_heavy_tail_demands(n_customers, rng)\n"
        "        elif demand_type == 'geo_correlated':\n"
        "            demands = generate_geo_correlated_demands(n_customers, coords, depot, rng)\n"
        "        else:\n"
        "            demands = generate_uniform_small_demands(n_customers, rng)\n"
        "        \n"
        "        regime_type = np.random.choice(regime_names, p=regime_weights)\n"
        "        if capacity is None:\n"
        "            total_demand = np.sum(demands)\n"
        "            if regime_type == 'loose':\n"
        "                m_target = max(3, round(n_customers / 6))\n"
        "                rho = 1.2\n"
        "            elif regime_type == 'medium':\n"
        "                m_target = max(3, round(n_customers / 7))\n"
        "                rho = 1.0\n"
        "            else:  # tight\n"
        "                m_target = max(3, round(n_customers / 8))\n"
        "                rho = 0.85\n"
        "            \n"
        "            Q = int(np.ceil(total_demand / m_target / rho))\n"
        "            Q = max(Q, int(np.max(demands)) + 1)  \n"
        "        else:\n"
        "            Q = capacity\n"
        "        \n"
        "        instance = {\n"
        "            'coords': coords,\n"
        "            'depot': depot,\n"
        "            'demands': demands,\n"
        "            'capacity': Q,\n"
        "            'n_customers': n_customers,\n"
        "            'seed': seed\n"
        "        }\n"
        "        instances.append(instance)\n"
        "    \n"
        "    return instances\n"
        "\n"
        "\n"
        "\n"
        "def generate_uniform_square(n, rng):\n"
        "    coords = rng.random((n, 2))\n"
        "    if rng.random() < 0.5:\n"
        "        depot = np.array([0.5, 0.5])\n"
        "    else:\n"
        "        depot = rng.random(2)\n"
        "    return coords, depot\n"
        "\n"
        "\n"
        "def generate_clustered_gaussian(n, rng):\n"
        "    k = rng.integers(3, 9)  # k=3~8\n"
        "    centers = rng.random((k, 2))\n"
        "    \n"
        "    sigma = rng.uniform(0.02, 0.08)\n"
        "    \n"
        "    coords = []\n"
        "    for i in range(n):\n"
        "        cluster_idx = rng.integers(0, k)\n"
        "        center = centers[cluster_idx]\n"
        "        point = center + rng.normal(0, sigma, size=2)\n"
        "        point = np.clip(point, 0.0, 1.0)\n"
        "        coords.append(point)\n"
        "    \n"
        "    coords = np.array(coords)\n"
        "    \n"
        "    if rng.random() < 0.5:\n"
        "        depot = centers[rng.integers(0, k)]\n"
        "    else:\n"
        "        depot = rng.random(2)\n"
        "    \n"
        "    return coords, depot\n"
        "\n"
        "\n"
        "def generate_two_scale_mixture(n, rng):\n"
        "    n_large = int(n * 0.7)\n"
        "    n_small = n - n_large\n"
        "    \n"
        "    coords = []\n"
        "    \n"
        "    for i in range(n_large):\n"
        "        coords.append(rng.random(2))\n"
        "    \n"
        "    corner_x = rng.choice([0.0, 1.0])\n"
        "    corner_y = rng.choice([0.0, 1.0])\n"
        "    cluster_size = rng.uniform(0.05, 0.15)\n"
        "    \n"
        "    for i in range(n_small):\n"
        "        if corner_x == 0.0:\n"
        "            x = rng.uniform(0.0, cluster_size)\n"
        "        else:\n"
        "            x = rng.uniform(1.0 - cluster_size, 1.0)\n"
        "        \n"
        "        if corner_y == 0.0:\n"
        "            y = rng.uniform(0.0, cluster_size)\n"
        "        else:\n"
        "            y = rng.uniform(1.0 - cluster_size, 1.0)\n"
        "        \n"
        "        coords.append([x, y])\n"
        "    \n"
        "    coords = np.array(coords)\n"
        "    indices = rng.permutation(n)\n"
        "    coords = coords[indices]\n"
        "    \n"
        "    if rng.random() < 0.5:\n"
        "        depot = np.array([0.5, 0.5])\n"
        "    else:\n"
        "        depot = rng.random(2)\n"
        "    \n"
        "    return coords, depot\n"
        "\n"
        "\n"
        "def generate_corridor(n, rng):\n"
        "    x_coords = rng.random(n)\n"
        "    y_coords = rng.uniform(0.0, 0.2, size=n)\n"
        "    coords = np.column_stack([x_coords, y_coords])\n"
        "    \n"
        "    angle = rng.uniform(0, 2 * np.pi)\n"
        "    cos_a = np.cos(angle)\n"
        "    sin_a = np.sin(angle)\n"
        "    rotation_matrix = np.array([[cos_a, -sin_a], [sin_a, cos_a]])\n"
        "    \n"
        "    coords = coords @ rotation_matrix.T\n"
        "    \n"
        "    coords_min = np.min(coords, axis=0)\n"
        "    coords_max = np.max(coords, axis=0)\n"
        "    coords_range = coords_max - coords_min\n"
        "    \n"
        "    if coords_range[0] > 0 and coords_range[1] > 0:\n"
        "        coords = (coords - coords_min) / coords_range\n"
        "    else:\n"
        "        coords = np.clip(coords, 0.0, 1.0)\n"
        "    \n"
        "    if rng.random() < 0.5:\n"
        "        depot = np.array([0.5, 0.5])\n"
        "    else:\n"
        "        depot = rng.random(2)\n"
        "    \n"
        "    return coords, depot\n"
        "\n"
        "\n"
        "def generate_annulus_ring(n, rng):\n"
        "    r0 = rng.uniform(0.2, 0.4)\n"
        "    r1 = rng.uniform(0.5, 0.7)\n"
        "    \n"
        "    coords = []\n"
        "    for i in range(n):\n"
        "        theta = rng.uniform(0, 2 * np.pi)\n"
        "        r = rng.uniform(r0, r1)\n"
        "        \n"
        "        x = r * np.cos(theta)\n"
        "        y = r * np.sin(theta)\n"
        "        \n"
        "        noise = rng.normal(0, 0.02, size=2)\n"
        "        x += noise[0]\n"
        "        y += noise[1]\n"
        "        \n"
        "        x = x + 0.5\n"
        "        y = y + 0.5\n"
        "        x = np.clip(x, 0.0, 1.0)\n"
        "        y = np.clip(y, 0.0, 1.0)\n"
        "        \n"
        "        coords.append([x, y])\n"
        "    \n"
        "    coords = np.array(coords)\n"
        "    \n"
        "    if rng.random() < 0.7:\n"
        "        depot = np.array([0.5, 0.5]) + rng.normal(0, 0.05, size=2)\n"
        "        depot = np.clip(depot, 0.0, 1.0)\n"
        "    else:\n"
        "        depot = rng.random(2)\n"
        "    \n"
        "    return coords, depot\n"
        "\n"
        "def generate_uniform_small_demands(n, rng):\n"
        "    \"\"\"uniform_small: qi ~ U{1..9}\"\"\"\n"
        "    demands = rng.integers(1, 10, size=n)\n"
        "    return demands\n"
        "\n"
        "\n"
        "def generate_bimodal_demands(n, rng):\n"
        "    n_small = n // 2\n"
        "    n_large = n - n_small\n"
        "    \n"
        "    small_demands = rng.integers(1, 5, size=n_small)\n"
        "    large_demands = rng.integers(6, 10, size=n_large)\n"
        "    \n"
        "    demands = np.concatenate([small_demands, large_demands])\n"
        "    indices = rng.permutation(n)\n"
        "    demands = demands[indices]\n"
        "    \n"
        "    return demands\n"
        "\n"
        "\n"
        "def generate_heavy_tail_demands(n, rng):\n"
        "    raw = rng.lognormal(mean=1.0, sigma=0.8, size=n)\n"
        "    \n"
        "    if raw.max() > raw.min():\n"
        "        demands = 1 + (raw - raw.min()) / (raw.max() - raw.min()) * 8\n"
        "    else:\n"
        "        demands = np.ones(n) * 5\n"
        "    \n"
        "    demands = np.round(demands).astype(int)\n"
        "    demands = np.clip(demands, 1, 9)\n"
        "    \n"
        "    return demands\n"
        "\n"
        "\n"
        "def generate_geo_correlated_demands(n, coords, depot, rng):\n"
        "    distances = np.linalg.norm(coords - depot, axis=1)\n"
        "    \n"
        "    if distances.max() > distances.min():\n"
        "        norm_distances = (distances - distances.min()) / (distances.max() - distances.min())\n"
        "    else:\n"
        "        norm_distances = np.zeros(n)\n"
        "    \n"
        "    a = 2.0\n"
        "    b = 5.0\n"
        "    \n"
        "    demands = a + b * norm_distances + rng.normal(0, 0.5, size=n)\n"
        "    demands = np.floor(demands).astype(int)\n"
        "    demands = np.clip(demands, 1, 9)\n"
        "    \n"
        "    return demands\n"
        "\n"
        "\n"
        "\n"
            "def apply_isometric_transform(coords, depot, rng):\n"
        "    if rng.random() < 0.5:\n"
        "        return coords, depot\n"
        "    \n"
        "    transform_type = rng.choice(['rotate', 'reflect', 'translate', 'scale'])\n"
        "    \n"
        "    if transform_type == 'rotate':\n"
        "        angle = rng.uniform(0, 2 * np.pi)\n"
        "        cos_a = np.cos(angle)\n"
        "        sin_a = np.sin(angle)\n"
        "        rotation_matrix = np.array([[cos_a, -sin_a], [sin_a, cos_a]])\n"
        "        \n"
        "        center = np.array([0.5, 0.5])\n"
        "        coords_centered = coords - center\n"
        "        depot_centered = depot - center\n"
        "        \n"
        "        coords_rotated = coords_centered @ rotation_matrix.T\n"
        "        depot_rotated = depot_centered @ rotation_matrix.T\n"
        "        \n"
        "        coords = coords_rotated + center\n"
        "        depot = depot_rotated + center\n"
        "        coords = np.clip(coords, 0.0, 1.0)\n"
        "        depot = np.clip(depot, 0.0, 1.0)\n"
        "        \n"
        "    elif transform_type == 'reflect':\n"
        "        axis = rng.choice([0, 1])\n"
        "        center = 0.5\n"
        "        \n"
        "        coords[:, axis] = 2 * center - coords[:, axis]\n"
        "        depot[axis] = 2 * center - depot[axis]\n"
        "        \n"
        "        coords = np.clip(coords, 0.0, 1.0)\n"
        "        depot = np.clip(depot, 0.0, 1.0)\n"
        "        \n"
        "    elif transform_type == 'translate':\n"
        "        translation = rng.uniform(-0.2, 0.2, size=2)\n"
        "        \n"
        "        coords = coords + translation\n"
        "        depot = depot + translation\n"
        "        \n"
        "        # Wrap到[0,1]^2\n"
        "        coords = coords % 1.0\n"
        "        depot = depot % 1.0\n"
        "        \n"
        "    elif transform_type == 'scale':\n"
        "        scale = rng.uniform(0.8, 1.0)  \n"
        "        center = np.array([0.5, 0.5])\n"
        "        \n"
        "        coords_centered = coords - center\n"
        "        depot_centered = depot - center\n"
        "        \n"
        "        coords_scaled = coords_centered * scale + center\n"
        "        depot_scaled = depot_centered * scale + center\n"
        "        \n"
        "        coords = np.clip(coords_scaled, 0.0, 1.0)\n"
        "        depot = np.clip(depot_scaled, 0.0, 1.0)\n"
        "    \n"
        "    return coords, depot\n"
    )

































