{
  "cells": [
    {
      "cell_type": "code",
      "execution_count": 5,
      "id": "a42b8ed0",
      "metadata": {},
      "outputs": [],
      "source": [
        "'''\n",
        "    The code for all our experiments is provided in this document.\n",
        "    We run it with Python 3.10.9 environment.\n",
        "    We provide detailed instructions on how to run the code with example usage.\n",
        "    We write the codes in a iPython notebook, all functions defined earlier are needed to run the final test.\n",
        "'''\n",
        "\n",
        "# c-juntas Hypothsis class\n",
        "\n",
        "import numpy as np\n",
        "from itertools import combinations\n",
        "from scipy.optimize import minimize\n",
        "\n",
        "def c_junta_hypothesis(d, c, num, coeff_range):\n",
        "    \"\"\"\n",
        "    Generate a list of c'-junta hyperplane hypotheses in d-dimensional space for all c' <= c.\n",
        "\n",
        "    Parameters:\n",
        "    d : int: Dimension of the input vector\n",
        "    c : int: Maximum number of active components (c <= d)\n",
        "    num : int: Number of discrete coefficient values\n",
        "    coeff_range : tuple: (lower_bound, upper_bound) for discretized coefficients\n",
        "\n",
        "    Returns:\n",
        "    hypotheses : list: List of callable hypothesis functions f(x) for all c'-juntas where c' <= c\n",
        "    \"\"\"\n",
        "    \n",
        "    (lb, ub) = coeff_range\n",
        "    coeff_values = np.linspace(lb, ub, num, dtype=float)\n",
        "    hypothesis_class = []\n",
        "\n",
        "    # Generate hypotheses for each c' from 1 to c\n",
        "    for c_prime in range(c, c + 1):\n",
        "        # Generate all possible subsets of c_prime indices\n",
        "        index_subsets = list(combinations(range(d), c_prime))\n",
        "        for indices in index_subsets:\n",
        "            # Generate all coefficient combinations for c_prime indices\n",
        "            for coeffs in np.array(np.meshgrid(*[coeff_values] * c_prime)).T.reshape(-1, c_prime):\n",
        "                def hypothesis(x, indices=indices, coeffs=coeffs):\n",
        "                    return np.sum(coeffs * x[list(indices)])\n",
        "                hypothesis_class.append({\n",
        "                    'indices': indices,\n",
        "                    'coefficients': coeffs,\n",
        "                    'hypothesis': hypothesis\n",
        "                })\n",
        "\n",
        "    return hypothesis_class\n",
        "\n",
        "def c_junta_hypothesis_fixed(d, c, num, coeff_range, indices):\n",
        "    \"\"\"\n",
        "    Generate a list of c'-junta hyperplane hypotheses in d-dimensional space for all c' <= c,\n",
        "    restricted to subsets of the given indices.\n",
        "\n",
        "    Parameters:\n",
        "    d : int: Dimension of the input vector\n",
        "    c : int: Maximum number of active components (c <= d)\n",
        "    num : int: Number of discrete coefficient values\n",
        "    coeff_range : tuple: (lower_bound, upper_bound) for discretized coefficients\n",
        "    indices : list or tuple: Specific set of indices to use for generating c'-junta hypotheses\n",
        "\n",
        "    Returns:\n",
        "    hypotheses : list: List of callable hypothesis functions f(x) for all c'-juntas where c' <= c\n",
        "    \"\"\"\n",
        "    \n",
        "    (lb, ub) = coeff_range\n",
        "    coeff_values = np.linspace(lb, ub, num, dtype=float)\n",
        "    hypothesis_class = []\n",
        "\n",
        "    # Generate hypotheses for each c' from 1 to c\n",
        "    for c_prime in range(c, c + 1):\n",
        "        # Generate all possible subsets of c_prime indices from the given indices\n",
        "        index_subsets = list(combinations(indices, c_prime))\n",
        "        for subset_indices in index_subsets:\n",
        "            # Generate all coefficient combinations for c_prime indices\n",
        "            for coeffs in np.array(np.meshgrid(*[coeff_values] * c_prime)).T.reshape(-1, c_prime):\n",
        "                def hypothesis(x, indices=subset_indices, coeffs=coeffs):\n",
        "                    return np.sum(coeffs * x[list(indices)])\n",
        "                hypothesis_class.append({\n",
        "                    'indices': subset_indices,\n",
        "                    'coefficients': coeffs,\n",
        "                    'hypothesis': hypothesis\n",
        "                })\n",
        "\n",
        "    return hypothesis_class\n",
        "\n",
        "def generate_random_hypothesis(d, c, coeff_range):\n",
        "    \"\"\"\n",
        "    Generate a random c-junta hypothesis in d-dimensional space.\n",
        "\n",
        "    Parameters:\n",
        "    d : int: Dimension of the input vector\n",
        "    c : int: Number of active components (c <= d)\n",
        "    coeff_range : tuple: (lower_bound, upper_bound) for random coefficients\n",
        "\n",
        "    Returns:\n",
        "    dict: Dictionary containing the indices, coefficients, and callable hypothesis function f(x)\n",
        "    \"\"\"\n",
        "    (lb, ub) = coeff_range\n",
        "    \n",
        "    # Randomly select c indices from {0, 1, ..., d-1}\n",
        "    indices = np.random.choice(d, c, replace=False)\n",
        "    \n",
        "    # Generate random coefficients within coeff_range for the c indices\n",
        "    coeffs = np.random.uniform(lb, ub, c)\n",
        "    \n",
        "    # Define the hypothesis function\n",
        "    def hypothesis(x, indices=indices, coeffs=coeffs):\n",
        "        return np.sum(coeffs * x[list(indices)])\n",
        "    \n",
        "    return hypothesis\n",
        "\n",
        "def c_junta_hypothesis_restricted(d, c, s, num, coeff_range):\n",
        "    \"\"\"\n",
        "    Generate a list of c-junta hyperplane hypotheses in d-dimensional space with restricted support.\n",
        "    For any subset of s indices out of c juntas, coefficients are discretized, and all other c-s coefficients are set to 1.\n",
        "    Ensures no two hypotheses have the same coefficients.\n",
        "\n",
        "    Parameters:\n",
        "    d : int: Dimension of the input vector\n",
        "    c : int: Number of active components (c <= d)\n",
        "    s : int: Number of components with discretized coefficients (s <= c)\n",
        "    num : int: Number of discrete coefficient values\n",
        "    coeff_range : tuple: (lower_bound, upper_bound) for discretized coefficients\n",
        "\n",
        "    Returns:\n",
        "    hypotheses : list: List of callable hypothesis functions f(x) with unique coefficients\n",
        "    \"\"\"\n",
        "    (lb, ub) = coeff_range\n",
        "    coeff_values = np.linspace(lb, ub, num, dtype=float)\n",
        "    index_subsets = list(combinations(range(d), c))\n",
        "    hypothesis_class = []\n",
        "    seen_coeffs = set()  # Track unique coefficient arrays\n",
        "\n",
        "    for indices in index_subsets:\n",
        "        # Generate all possible s-subsets from the c indices\n",
        "        s_subsets = list(combinations(indices, s))\n",
        "        for s_indices in s_subsets:\n",
        "            # Remaining c-s indices have coefficient 1\n",
        "            fixed_indices = [i for i in indices if i not in s_indices]\n",
        "            for coeffs in np.array(np.meshgrid(*[coeff_values] * s)).T.reshape(-1, s):\n",
        "                # Create full coefficient array: discretized for s_indices, 1 for others\n",
        "                full_coeffs = np.ones(c) * coeff_values[num // 2]\n",
        "                # Map s_indices to their positions in the full c indices\n",
        "                s_positions = [list(indices).index(i) for i in s_indices]\n",
        "                for pos, coeff in zip(s_positions, coeffs):\n",
        "                    full_coeffs[pos] = coeff\n",
        "                # Convert to tuple for hashing\n",
        "                coeff_tuple = tuple(full_coeffs)\n",
        "                if coeff_tuple not in seen_coeffs:\n",
        "                    seen_coeffs.add(coeff_tuple)\n",
        "                    def hypothesis(x, indices=indices, coeffs=full_coeffs):\n",
        "                        return np.sum(coeffs * x[list(indices)])\n",
        "                    hypothesis_class.append({\n",
        "                        'indices': indices,\n",
        "                        'coefficients': full_coeffs,\n",
        "                        'hypothesis': hypothesis\n",
        "                    })\n",
        "\n",
        "    return hypothesis_class\n",
        "\n",
        "def get_callable_hypothesis(hypothesis_class):\n",
        "    \"\"\" Generate a list of callable hypothesis\"\"\"\n",
        "\n",
        "    callable_functions = []\n",
        "    for hypothesis in hypothesis_class:\n",
        "        callable_functions.append(hypothesis['hypothesis'])\n",
        "    return callable_functions"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 6,
      "id": "e51d25a7",
      "metadata": {
        "id": "e51d25a7"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "\"\\nhypothesis_class = get_callable_hypothesis(c_junta_hypothesis(5, 5, 6, (0,1)))\\n# Generate sequences once per iteration for consistency across m values\\ninput_dict = {'dimension': 5, 'number_of_rounds': 100, 'range_of_coefficient': (0,1), 'support_of_example': 2, 'dynamical_system': dynamical_system_basic}\\nx_sequence = set_input(input_dict)\\nhypothesis = np.random.choice(hypothesis_class)\\ny_true, y_sequence = generate_labels(x_sequence, hypothesis)\\nprint('x_sequence:', x_sequence)\\nprint('y_sequence:', y_sequence)\\n\""
            ]
          },
          "execution_count": 6,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "# Basis functions\n",
        "\n",
        "def dynamical_system_basic(x0, A, T, d):\n",
        "    \"\"\"\n",
        "    Simulate the dynamical system x_{t+1} = Ax_t.\n",
        "\n",
        "    Parameters:\n",
        "    x0 : ndarray: Initial state vector of shape (d,)\n",
        "    A : ndarray: Transition matrix of shape (d, d)\n",
        "    T : int: Number of time steps\n",
        "    d : int: Dimension of the state vector\n",
        "\n",
        "    Returns:\n",
        "    x_sequence : Array of shape (T + 1, d) containing the sequence x_0, x_1, ..., x_T\n",
        "    \"\"\"\n",
        "\n",
        "    # Initialize output array\n",
        "    x_sequence = np.zeros((T + 1, d))\n",
        "    x_sequence[0] = x0\n",
        "\n",
        "    # Compute the sequence iteratively\n",
        "    for t in range(T):\n",
        "        x_sequence[t + 1] = A @ x_sequence[t]\n",
        "    x_sequence[x_sequence == 0] = 0\n",
        "\n",
        "    return x_sequence\n",
        "\n",
        "def generate_random_stable_matrix(d):\n",
        "    \"\"\"\n",
        "    Generate a random stable matrix A of shape (d, d) with eigenvalues |λ| < 1.\n",
        "\n",
        "    Parameters:\n",
        "        d: Dimension of the square matrix\n",
        "\n",
        "    Returns:\n",
        "        A: Stable matrix of shape (d, d)\n",
        "    \"\"\"\n",
        "    # Generate random matrix\n",
        "    A = np.random.randn(d, d)\n",
        "\n",
        "    # Compute eigenvalues and eigenvectors\n",
        "    eigenvalues, eigenvectors = np.linalg.eig(A)\n",
        "\n",
        "    # Scale eigenvalues to ensure |λ| < 1 for stability\n",
        "    eigenvalues = 0.999 * eigenvalues / (np.abs(eigenvalues) + 1e-100)\n",
        "\n",
        "    # Reconstruct matrix: A = V * diag(λ) * V⁻¹\n",
        "    A = eigenvectors @ np.diag(eigenvalues) @ np.linalg.inv(eigenvectors)\n",
        "\n",
        "    #print(A)\n",
        "\n",
        "    return A\n",
        "\n",
        "def set_input(input, non_zero_indices):\n",
        "    \"\"\"\n",
        "    Set the input of the problem.\n",
        "\n",
        "    Parameters:\n",
        "    input: dictionary: The input parameters and the choice of the dynamical system\n",
        "\n",
        "    Returns:\n",
        "    x_sequence : Array of shape (T + 1, d) containing the sequence x_0, x_1, ..., x_T\n",
        "\n",
        "    \"\"\"\n",
        "    d = input['dimension']\n",
        "    T = input['number_of_rounds']\n",
        "    s = input['support_of_example']\n",
        "    dynamical_system = input['dynamical_system']\n",
        "\n",
        "    #Generate the input\n",
        "    x0 = np.zeros(d)\n",
        "    x0[non_zero_indices] = np.random.normal(0, 1, size = s)\n",
        "    A = np.zeros((d, d))\n",
        "    submatrix = generate_random_stable_matrix(s)\n",
        "    A[np.ix_(non_zero_indices, non_zero_indices)] = submatrix\n",
        "\n",
        "    # Run simulation\n",
        "    return dynamical_system(x0, A, T, d)\n",
        "\n",
        "# Example: input = {'dimension': 4, 'number_of_rounds': 10, 'dynamical_system': dynamical_system_basic}\n",
        "\n",
        "def generate_labels(x_sequence, hypothesis, noise_std = 0.1):\n",
        "    \"\"\"\n",
        "    Map state sequence x_t to labels y_t = f(x_t) + n_t, where n_t is Gaussian noise.\n",
        "\n",
        "    Parameters:\n",
        "    x_sequence : ndarray: State sequence of shape (T+1, d) from the dynamical system\n",
        "    f : callable: Function to map x_t to y_t (before noise), takes a vector of shape (d,) and returns a scalar or vector\n",
        "    noise_std : float: Standard deviation of Gaussian noise (default: 0.1)\n",
        "\n",
        "    Returns:\n",
        "    y_sequence : ndarray: Sequence of labels y_0, y_1, ..., y_T, shape depends on f's output\n",
        "    \"\"\"\n",
        "    T, d = x_sequence.shape\n",
        "    # Apply f to each state vector\n",
        "    y_true = np.array([hypothesis(x_sequence[t]) for t in range(T)])\n",
        "\n",
        "    # Add Gaussian noise\n",
        "    noise = np.random.normal(0, noise_std, y_true.shape)\n",
        "    y_sequence = y_true + noise\n",
        "\n",
        "    return y_true, y_sequence\n",
        "\n",
        "def l1_loss(y_true, y_est):\n",
        "    \"\"\"\n",
        "    Compute the L1 loss (mean absolute error) between true and predicted label sequences.\n",
        "\n",
        "    Parameters:\n",
        "    y_true : ndarray: True label sequence of shape (T+1,) or (T+1, k)\n",
        "    y_est : ndarray: Predicted label sequence of shape (T+1,) or (T+1, k)\n",
        "\n",
        "    Returns:\n",
        "    loss : float: L1 loss (mean absolute error) over the sequence\n",
        "    \"\"\"\n",
        "\n",
        "    return np.sum(np.abs(y_true - y_est))\n",
        "\n",
        "'''\n",
        "hypothesis_class = get_callable_hypothesis(c_junta_hypothesis(5, 5, 6, (0,1)))\n",
        "# Generate sequences once per iteration for consistency across m values\n",
        "input_dict = {'dimension': 5, 'number_of_rounds': 100, 'range_of_coefficient': (0,1), 'support_of_example': 2, 'dynamical_system': dynamical_system_basic}\n",
        "x_sequence = set_input(input_dict)\n",
        "hypothesis = np.random.choice(hypothesis_class)\n",
        "y_true, y_sequence = generate_labels(x_sequence, hypothesis)\n",
        "print('x_sequence:', x_sequence)\n",
        "print('y_sequence:', y_sequence)\n",
        "'''"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 7,
      "id": "c8c446c0",
      "metadata": {
        "id": "c8c446c0"
      },
      "outputs": [],
      "source": [
        "# Multiplicative Weight\n",
        "import random\n",
        "\n",
        "def multiplicative_weight_regression(x_sequence, y_sequence,\n",
        "                                  net,\n",
        "                                  loss,\n",
        "                                  eta = 0.1):\n",
        "    \"\"\"\n",
        "    Run Multiplicative Weight Method for online regression.\n",
        "\n",
        "    Parameters:\n",
        "        x_sequence: List of input data points\n",
        "        y_sequence: List of true labels\n",
        "        hypothesis_class: List of hypothesis functions\n",
        "        loss: Loss function taking (prediction, true_label) as input\n",
        "        eta: Learning rate parameter\n",
        "\n",
        "    Returns:\n",
        "        List of predictions\n",
        "    \"\"\"\n",
        "    n_hypotheses = len(net)\n",
        "    weights = np.ones(n_hypotheses) / n_hypotheses\n",
        "    cumulative_losses = np.zeros(n_hypotheses)\n",
        "    y_estimate = []\n",
        "\n",
        "    for x, y in zip(x_sequence, y_sequence):\n",
        "        # Sample hypothesis index based on weights\n",
        "        idx = np.random.choice(n_hypotheses, p = weights)\n",
        "        #print(idx)\n",
        "        prediction = net[idx](x)\n",
        "        y_estimate.append(prediction)\n",
        "\n",
        "        # Calculate loss for each hypothesis\n",
        "        for i in range(n_hypotheses):\n",
        "            prediction = net[i](x)\n",
        "            cumulative_losses[i] += loss(prediction, y)\n",
        "\n",
        "        # Update weights\n",
        "        weights = np.exp(-eta * cumulative_losses)\n",
        "        weights /= np.sum(weights)  # Normalize probabilities\n",
        "        #print(weights)\n",
        "\n",
        "    return y_estimate\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 11,
      "id": "17ae2168",
      "metadata": {},
      "outputs": [
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "/var/folders/43/yrldk00d1t7dhdpl_hgvd2jh0000gn/T/ipykernel_44305/1248789593.py:75: ComplexWarning: Casting complex values to real discards the imaginary part\n",
            "  A[np.ix_(non_zero_indices, non_zero_indices)] = submatrix\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Number of hypotheses in full hypothesis class: 2160\n",
            "Number of hypotheses in restricted hypothesis class: 216\n"
          ]
        },
        {
          "data": {
            "image/png": "iVBORw0KGgoAAAANSUhEUgAAA1IAAAIjCAYAAAAJLyrXAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjAsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvlHJYcgAAAAlwSFlzAAAPYQAAD2EBqD+naQAAmyVJREFUeJzs3Xd8U9Ubx/FPugdtoVAoe+8te++NAjJdDAcOFBABRQUZIrIUURQnoP4UFQQFEahM2UtQhgjI3rusztzfH8emFAq00DZp+32/Xnk1Offm3ifJRfPknPMcm2VZFiIiIiIiIpJkbs4OQEREREREJL1RIiUiIiIiIpJMSqRERERERESSSYmUiIiIiIhIMimREhERERERSSYlUiIiIiIiIsmkREpERERERCSZlEiJiIiIiIgkkxIpERERERGRZFIiJSLiwg4cOIDNZmP69OnODsWhYcOGNGzYMFXPUahQIXr27Jmq55DEJXbNDR8+HJvNlqrn7dmzJ4UKFUrVc4iIpCQlUiKS6UyfPh2bzea4eXh4kDdvXnr27MnRo0dT7bwLFixg+PDhqXb8661Zs4bhw4dz4cKFNDlferRz506GDx/OgQMHnB3KPVu+fHmCa/r6W7du3Zwdnsu4evUqw4cPZ/ny5c4ORUQyAA9nByAi4iwjR46kcOHCREREsG7dOqZPn86qVavYvn07Pj4+KX6+BQsWMGXKlGQlUwULFuTatWt4enom61xr1qxhxIgR9OzZk6xZsyYvUBewe/du3NxS97e+nTt3MmLECBo2bJhhekL69u1LtWrVErSll9f26aefYrfbU/UcV69eZcSIEQCp3qsqIhmfEikRybRatWpF1apVAXjyySfJkSMHY8eO5eeff6ZLly5OjS0mJga73Y6Xl1eqJHWuztvb29khpEv16tWjU6dOzg7jriT3xwIREWfT0D4Rkf/Uq1cPgH379iVo//vvv+nUqRPBwcH4+PhQtWpVfv755wT7REdHM2LECIoXL46Pjw/Zs2enbt26hIWFAWb+x5QpUwASDLuC+DkpEyZMYNKkSRQtWhRvb2927tx5yzlSf//9N126dCEkJARfX19KlizJa6+9Bpj5LIMGDQKgcOHCjnNdP4Tt66+/pkqVKvj6+hIcHEy3bt04fPjwTe/JJ598QtGiRfH19aV69er8/vvvyXpPv/76a6pXr46fnx/ZsmWjfv36LF68+I7Pu3GO1K3m6MQN07z+tRUqVIi2bduyatUqqlevjo+PD0WKFOHLL79M8LzOnTsD0KhRI8d7FDfk66effqJNmzbkyZMHb29vihYtyqhRo4iNjb1t3LNmzcJms7FixYqbtn388cfYbDa2b98OwIkTJ+jVqxf58uXD29ub3Llz065du1QbanireWcpPedt/fr1tG7dmmzZsuHv70+FChV477337vi8G+dIxQ1XvHEYXmL/Jnr27EmWLFk4evQo7du3J0uWLISEhDBw4EDHZ3bgwAFCQkIAGDFihOMzj+sh/vPPP+nZsydFihTBx8eH0NBQHn/8cc6ePXtP74eIZFzqkRIR+U/cF9hs2bI52nbs2EGdOnXImzcvr7zyCv7+/nz//fe0b9+e2bNn06FDB8B80R8zZgxPPvkk1atXJzw8nE2bNrFlyxaaNWvG008/zbFjxwgLC+Orr75K9PzTpk0jIiKC3r174+3tTXBwcKJDnf7880/q1auHp6cnvXv3plChQuzbt4958+YxevRoHnzwQf755x++/fZb3n33XXLkyAHg+BI5evRohg4dSpcuXXjyySc5ffo077//PvXr1+ePP/5wDAX8/PPPefrpp6lduzb9+/fn33//5YEHHiA4OJj8+fPf8f0cMWIEw4cPp3bt2owcORIvLy/Wr1/P0qVLad68eZI/l7uxd+9eOnXqxBNPPEGPHj344osv6NmzJ1WqVKFs2bLUr1+fvn37MnnyZF599VVKly4N4Pg7ffp0smTJwoABA8iSJQtLly5l2LBhhIeHM378+Fuet02bNmTJkoXvv/+eBg0aJNj23XffUbZsWcqVKwdAx44d2bFjBy+88AKFChXi1KlThIWFcejQobsejnfp0iXOnDmToC04ODjVh0nGCQsLo23btuTOnZt+/foRGhrKrl27mD9/Pv369UvVc8fGxtKiRQtq1KjBhAkT+O2335g4cSJFixbl2WefJSQkhI8++ohnn32WDh068OCDDwJQoUIFR+z//vsvvXr1IjQ0lB07dvDJJ5+wY8cO1q1bl+rFNkQkHbJERDKZadOmWYD122+/WadPn7YOHz5szZo1ywoJCbG8vb2tw4cPO/Zt0qSJVb58eSsiIsLRZrfbrdq1a1vFixd3tFWsWNFq06bNbc/bp08fK7H/7O7fv98CrMDAQOvU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0qFDB/LkycPevXuZOnUqdevWpV+/fjc9r3jx4nz11VdUrVqV8PBwBg0ahK+vr2P79OnTiY2NpUaNGvj5+fH111/j6+tLwYIFmT9/Pv/++y9169bFw8OD33//HbvdTsmSJe/6dYiIiIgkx4kTsG4drF0LFy/Chg2wcydcV9MtgUKFoHRpKFUq/m+pUpA1awwLFqyidevWaRr/vXJqIrVp0yZH2WrAMW+pR48eTJ8+ncGDB3PlyhV69+7NhQsXqFu3LgsXLkwwV+J///sfzz//PE2aNMHNzY2OHTsyefLkNH8t6UmLFi0YOnQogwcPJiIigscff5zu3bs7ejpSWunSpfnwww956623GDVqFB07dmTgwIF88sknyTrORx99xKuvvspzzz3H2bNnKVCgAK+++ioAefPmZcSIEbzyyiv06tWL7t27M3369JuOMW3aNPr160fbtm2Jioqifv36LFiw4J6HHowdO5YqVaowZcoUpk6dit1up2jRonTq1OmW5c8///xzevfuzX333Uf+/Pl56623GDhwoGN71qxZefvttxkwYACxsbGUL1+eefPmkT17drJmzcqPP/7I8OHDiYiIoHjx4nz77beULVv2nl6HiIiIyI0syyRNO3fC1q2wfr1JoA4fTnx/X18oUwZKlIBq1aBKFahUCW71G7qLFOtLNptlJXd1oownPDycoKAgLl68mGixif3791O4cOFUn+weV8UuMDDwnudISeaQ0a+ZtPz3l1lER0ezYMECWrdurbkLkiS6ZiS5dM2kX5YFx47B/v1m7tLOnbBsmfl78WLizylXDmrWhDx5TOJUqxYULAjJWbXE1a6Z2+UG13PZYhMiIiIiIpLy7HaTMP37r0mS/vor/naLqd24uUGRIqanqXZt08t0330QHJymobsUJVIiIiIiIhnUhQtmDtP27bBtm6mUt337recxububHqUCBcycpoYNTcJUvDhocEhCSqRERERERDKA8+dNshR327zZ9DIlNpEnLmEqWRLKl4+/lSoFiawSJIlQIiUiIiIiks6cPw9//AHLl5sCEFu33rr4Q7FiULVqfAGIqlVNEuWhTOCe6O0TEREREXFhZ8+aSnl//GGG6f37L/zzj1mz6UaFCkHFivG32rUhNDTNQ84UlEiJiIiIiLgIy4KjR808ps2b4YcfYMcOiIm5ed8CBaBePahRw5QX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hF69zNpOmuskIumREikRERG5a5cvw+rVsHatqbC3bp3piYpjs0HBgvDAA2aeU40aJpESEUnvlEiJiIhIssTEmDlOixbB99/DuXMJt1esaApENGtmKu35+zsnThGR1KRESkRERJJk92746iuYMSPhkL18+aBhQ5M03X+/epxEJHNQIiUiIiKJunQJFiyw8emnlRg61IO//orfFhICDz8MjRpBmzbgoW8UIpLJ6D97IiIi4nDggKmwN28eLF8O0dEeQEEA3N2hRQtTYa99e1XZE5HMTYmUiIhIJhYVZQpF/PADLFsGO3cm3F6smEWZMvt49NFC1K/vQa5czolTRMTVKJESERHJZCIjTXW9hQth6lS4cCF+m7s71K0Lbdua+U5FisSwYMEOWrcuiKen00IWEXE5SqREREQyuNhY2LIFVq6EVavM+k5XrsRv9/eHzp3NXKfGjSE4OH5bdHTaxysikh4okRIREcmAIiLMPKfvvzdznc6cSbg9Vy5Taa9TJ+jQwfREiYhI0imREhERyUC2boXPP4f//Q/On49vDwqCypWhVSvT61SlilksV0RE7o4SKRERkXTMsmD1avjxR/j1V/j77/ht+fKZEuXt2pnEydvbeXGKiGQ0SqRERETSoSNHTOL06aewcWN8u6enKU3++OPQrJmG7ImIpBYlUiIiIulATAwsWmRKlK9cCZs3g91utvn4QNeuplhE06aQLZtzYxURyQyUSImIiLioqChYsQIWL4aZM00v1PVq1IDWreGZZyBnTufEKCKSWSmREhERcSHR0aZQxLx5pkz5pUvx20JCzLC9Bg2gXj0oUMBpYYqIZHpKpERERFzAihUwYQIsXQpXr8a3h4aaXqfmzU3RCB8f58UoIiLxlEiJiIg4ycmTJnGaPt0M34uTPTs895yZ81StGri5OS1EERG5BSVSIiIiacSyYNs2+OknmD8fNm2K3+bhAU89ZeY7lSun5ElExNUpkRIREUllJ06YeU8zZsBffyXcVrmyGbb31FNQtKhz4hMRkeRTIiUiIpIKjhyB7783BSPCwiA21rR7e0OrVvDAA+ZvaKhz4xQRkbvj0gMHYmNjGTp0KIULF8bX15eiRYsyatQoLMty7GNZFsOGDSN37tz4+vrStGlT9uzZ48SoRUQkM7Is0/O0YAG0aGEq6r30EixcaJKomjVh6lQ4fhzmzIFe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FBSkXr16SZIWL16snJwcPfXUUyopKdHjjz+u7OxsrVy50mM7L730kgYNGqStW7eqX79+ysjIUGVl5XnrX716tVJSUi7p2Lt166bVq1df0roAgPpopAAAV1WXLl303HPPqX379ho/fryCgoIUHR2tYcOGqX379powYYJ+/PFHbd26VZL017/+VcnJyZo0aZI6duyo5ORkzZ49WytXrtSuXbsuuK82bdooNDTU41FaWupeXlBQoG3btum9995T165d1b17d7399ttatWqV+zI46dTlfG+//baSk5N1yy231NtPQECA+6yPv7+/fv/732vo0KEaOnSoJGnq1KnKysrSE088oRtvvFF5eXm6//77NXXqVI/tZGVlafDgwWrXrp0mTZqkI0eOqKio6LzHt3fvXsXGxv73F/0cYmNjVVZWxu+kAOAKoZECAFxVZzYivr6+ioqKUlJSknvM6XRKkg4cOCBJ2rJli1auXOnRDHXs2FGS9N13311wX6tXr1ZxcbHH48zGY8eOHYqLi1NcXJx7rFOnToqMjNSOHTvcY23btnVfFnchtbW1euCBB9S2bVvNmDHDYz89e/b0mNuzZ0+PfUier02zZs0UHh7ufh3O5dixY+5LDa0KDg6Wy+XS8ePHL2l9AIAnP7sLAABc2/z9/T2eOxwOjzGHwyFJ7jMlR44c0b333qs//elP9bbVqlWrC+4rMTGx3s0u/Pysf9Q1a9bsouaNGDFCZWVlKioquqT9nOu1udAZo+joaB06dMjyfiSpsrJSzZo1U3Bw8CWtDwDwRCMFAPAqt912mxYuXKiEhIRLak4u5KabblJZWZnKysrcZ6W2b9+uqqoqderUydK2pk2bpvnz5+vLL79UVFRUvf2sXbtWmZmZ7rG1a9da3sfZkpOTtX379ktat6SkRMnJyZe1fwDAz7i0DwDgVUaOHKnKykoNHjxY69ev13fffafly5crOztbdXV1l7XttLQ0JSUlKSMjQ5s2bVJRUZEeffRR3XnnnZZu4vDZZ5/pmWee0SuvvKLo6GiVl5ervLxchw8fliSNGTNGc+fO1axZs7R7925NmzZNixYt0tNPP31Z9ffp00eFhYWX9DqsXr1a6enpl7V/AMDPaKQAAF4lNjZWa9euVV1dndLT05WUlKTc3FxFRkbKx+fyPrYcDoc++ugjNW/eXL169VJaWpquv/56ffDBB5a2s2bNGtXV1Wn48OFq1aqV+5GTkyNJGjhwoGbMmKGpU6fq5ptv1uuvv645c+aod+/el1V/37595efnp88++8zSevv27dOXX36p7Ozsy9o/AOBnDmOMsbsIAABwcWbOnKklS5Zo+fLlF73O2LFjdejQIb3xxhtXsTIAaFr4jRQAAI3I448/rqqqKtXU1CgsLOyi1mnZsqXy8vKucmUA0LRwRgoAAAAALOI3UgAAAABgEY0UAAAAAFhEIwUAAAAAFtFIAQAAAIBFNFIAAAAAYBGNFAAAAABYRCMFAAAAABbRSAEAAACARTRSAAAAAGDR/wGEMQesrkwLjQAAAABJRU5ErkJggg==",
            "text/plain": [
              "<Figure size 1000x600 with 1 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        }
      ],
      "source": [
        "# Comparison - restricted indice\n",
        "import matplotlib.pyplot as plt\n",
        "\n",
        "def main_multi_comparison(input_dict, loss, iter):\n",
        "    \"\"\"\n",
        "    Run multiplicative weight regression for the full c-junta hypothesis class\n",
        "    and a restricted c-junta hypothesis class with a specific randomly chosen support,\n",
        "    and compare mean L1 loss. Track cumulative loss over time for plotting.\n",
        "\n",
        "    Args:\n",
        "        x_sequence: ndarray: Array of input points (T x d)\n",
        "        input_dict: Dictionary with 'dimension', 'number_of_rounds', 'support_of_example', \n",
        "                    'dynamical_system', 'juntas', 'choices_of_coefficients', 'range_of_coefficient'\n",
        "        loss: Loss function (e.g., l1_loss)\n",
        "        iter: Number of iterations to run for averaging\n",
        "        dist: Distance threshold for constructing restricted net\n",
        "\n",
        "    Returns:\n",
        "        Dictionary with mean L1 losses and cumulative loss arrays for plotting\n",
        "    \"\"\"\n",
        "    # Initialize arrays to store losses across iterations\n",
        "    full_hyp_losses = np.zeros(iter)\n",
        "    restricted_hyp_losses = np.zeros(iter)\n",
        "    d = input_dict['dimension']\n",
        "    c = input_dict['juntas']\n",
        "    s = input_dict['support_of_example']\n",
        "    T = input_dict['number_of_rounds']\n",
        "    num = input_dict['choices_of_coefficients']\n",
        "    coeff_range = input_dict['range_of_coefficient']\n",
        "    \n",
        "    # Initialize arrays to store cumulative losses across iterations\n",
        "    full_hyp_cum_losses = np.zeros((iter, T))\n",
        "    restricted_hyp_cum_losses = np.zeros((iter, T))\n",
        "    \n",
        "    # Define the full hypothesis class (all c-juntas)\n",
        "    hypo = c_junta_hypothesis(d, c, num, coeff_range)\n",
        "    #for hyp in hypo: print(hyp)\n",
        "    full_hypothesis_class = get_callable_hypothesis(hypo)\n",
        "    \n",
        "    # Randomly select c indices as the specific support for restricted class\n",
        "    specific_indices = np.random.choice(d, c, replace=False).tolist()\n",
        "    \n",
        "    x_sequence = set_input(input_dict, specific_indices)\n",
        "    #print(x_sequence)\n",
        "\n",
        "    # Define the restricted hypothesis class (c-juntas with specific indices)\n",
        "    hypo_restricted = c_junta_hypothesis_fixed(d, c, num, coeff_range, specific_indices)\n",
        "    #for hyp in hypo_restricted: print(hyp)\n",
        "    restricted_hypothesis_class = get_callable_hypothesis(hypo_restricted)\n",
        "    \n",
        "    for i in range(iter):\n",
        "        # Generate target hypothesis and labels\n",
        "        target_hypothesis = np.random.choice(restricted_hypothesis_class)\n",
        "        #target_hypothesis = generate_random_hypothesis(d, c, coeff_range)\n",
        "        y_true, y_sequence = generate_labels(x_sequence, target_hypothesis)\n",
        "        \n",
        "        # Run MWU for full hypothesis class\n",
        "        y_est_full = multiplicative_weight_regression(x_sequence, y_sequence, full_hypothesis_class, loss, 0.1)\n",
        "        full_losses = np.array([loss(np.array([y_true[t]]), np.array([y_est_full[t]])) for t in range(T)])\n",
        "        full_hyp_cum_losses[i] = np.cumsum(full_losses)\n",
        "        full_hyp_losses[i] = full_losses[-1]\n",
        "        \n",
        "        # Run MWU for restricted hypothesis class\n",
        "        y_est_junta = multiplicative_weight_regression(x_sequence, y_sequence, restricted_hypothesis_class, loss, 0.1)\n",
        "        junta_losses = np.array([loss(np.array([y_true[t]]), np.array([y_est_junta[t]])) for t in range(T)])\n",
        "        restricted_hyp_cum_losses[i] = np.cumsum(junta_losses)\n",
        "        restricted_hyp_losses[i] = junta_losses[-1]\n",
        "        \n",
        "        # Print hypothesis class sizes (once, for first iteration)\n",
        "        if i == 0:\n",
        "            print(f\"Number of hypotheses in full hypothesis class: {len(full_hypothesis_class)}\")\n",
        "            print(f\"Number of hypotheses in restricted hypothesis class: {len(restricted_hypothesis_class)}\")\n",
        "    \n",
        "    # Compute mean cumulative losses across iterations\n",
        "    full_mean_cum_loss = np.mean(full_hyp_cum_losses, axis=0)\n",
        "    restricted_mean_cum_loss = np.mean(restricted_hyp_cum_losses, axis=0)\n",
        "    \n",
        "    # Compute mean losses across iterations\n",
        "    full_mean_loss = np.mean(full_hyp_losses)\n",
        "    restricted_mean_loss = np.mean(restricted_hyp_losses)\n",
        "    \n",
        "    # Plot the results\n",
        "    plt.figure(figsize=(10, 6))\n",
        "    plt.plot(range(1, T + 1), full_mean_cum_loss, label='Full Function Class', color='blue')\n",
        "    plt.plot(range(1, T + 1), restricted_mean_cum_loss, label='Restricted Function Class', color='red')\n",
        "    plt.xlabel('Time Horizon (T)')\n",
        "    plt.ylabel('Total L1 Loss')\n",
        "    plt.title('Restricted c-junta vs. Full c-junta')\n",
        "    plt.legend()\n",
        "    plt.grid(True)\n",
        "    plt.savefig('loss_comparison.png')\n",
        "    plt.show()\n",
        "    \n",
        "    return {\n",
        "        'full_hyp_mean_loss': full_mean_loss.item(),\n",
        "        'junta_hyp_mean_loss': restricted_mean_loss.item(),\n",
        "        'full_hyp_cum_losses': full_mean_cum_loss,\n",
        "        'junta_hyp_cum_losses': restricted_mean_cum_loss\n",
        "    }\n",
        "\n",
        "# Example usage\n",
        "input_dict = {'dimension': 8, 'juntas': 4, 'number_of_rounds': 1000, 'range_of_coefficient': (-1,1), \n",
        "              'choices_of_coefficients': 6, 'support_of_example': 4, 'dynamical_system': dynamical_system_basic}\n",
        "result = main_multi_comparison(input_dict, loss = l1_loss, iter = 5)"
      ]
    }
  ],
  "metadata": {
    "colab": {
      "provenance": []
    },
    "kernelspec": {
      "display_name": "Python 3 (ipykernel)",
      "language": "python",
      "name": "python3"
    },
    "language_info": {
      "codemirror_mode": {
        "name": "ipython",
        "version": 3
      },
      "file_extension": ".py",
      "mimetype": "text/x-python",
      "name": "python",
      "nbconvert_exporter": "python",
      "pygments_lexer": "ipython3",
      "version": "3.10.9"
    }
  },
  "nbformat": 4,
  "nbformat_minor": 5
}
