{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "44282f4e",
   "metadata": {},
   "source": [
    "# load config"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 88,
   "id": "8891e07a",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import re\n",
    "import pandas as pd\n",
    "\n",
    "# model sizes\n",
    "\n",
    "mixtral_model_config = {'15m': 0.09218912,\n",
    " '44m': 0.27653808,\n",
    " '0.1b': 0.61449664,\n",
    " '0.2b': 1.453845952,\n",
    " '0.5b': 3.26173232,\n",
    " '1b': 6.786500704}\n",
    "\n",
    "mixtral_total_model_activeparam_config = {'15m': 0.04629664,\n",
    " '44m': 0.10815792,\n",
    " '0.1b': 0.21303104,\n",
    " '0.2b': 0.460346432,\n",
    " '0.5b': 0.9850008,\n",
    " '1b': 1.996531424}\n",
    "\n",
    "mixtral_4x_model_config= {'15m': 0.0479408,\n",
    " '44m': 0.14380464,\n",
    " '0.1b': 0.31954624,\n",
    " '0.2b': 0.756014272,\n",
    " '0.5b': 1.69612464,\n",
    " '1b': 3.529017184}\n",
    "\n",
    "mixtral_16x_model_config = {'15m': 0.18068576,\n",
    " '44m': 0.54200496,\n",
    " '0.1b': 1.20439744,\n",
    " '0.2b': 2.849509312,\n",
    " '0.5b': 6.39294768,\n",
    " '1b': 13.301467744}\n",
    "\n",
    "llama_model_config = {'15m': 0.01475168,\n",
    " '44m': 0.0442488,\n",
    " '0.1b': 0.09832384,\n",
    " '0.2b': 0.23262304,\n",
    " '0.5b': 0.52188976,\n",
    " '1b': 1.085859424}\n",
    "\n",
    "mixtral_model_activeparam_config = {'15m': 0.02581664,\n",
    " '44m': 0.07743792,\n",
    " '0.1b': 0.17207104,\n",
    " '0.2b': 0.407098432,\n",
    " '0.5b': 0.9133208,\n",
    " '1b': 1.900275424}\n",
    "\n",
    "mixtral_model_topk_1_activeparam_config= {'15m': 0.01475456,\n",
    " '44m': 0.04425456,\n",
    " '0.1b': 0.09833344,\n",
    " '0.2b': 0.232640512,\n",
    " '0.5b': 0.52191888,\n",
    " '1b': 1.085904544}\n",
    "\n",
    "mixtral_4x_topk_1_activeparam_config = {'15m': 0.01475456,\n",
    " '44m': 0.04425456,\n",
    " '0.1b': 0.09833344,\n",
    " '0.2b': 0.232640512,\n",
    " '0.5b': 0.52191888,\n",
    " '1b': 1.085904544}\n",
    "\n",
    "mixtral_4x_topk_2_activeparam_config = {'15m': 0.02581664,\n",
    " '44m': 0.07743792,\n",
    " '0.1b': 0.17207104,\n",
    " '0.2b': 0.407098432,\n",
    " '0.5b': 0.9133208,\n",
    " '1b': 1.900275424}\n",
    "\n",
    "mixtral_16x_topk_2_activeparam_config = {'15m': 0.18068576,\n",
    " '44m': 0.54200496,\n",
    " '0.1b': 1.20439744,\n",
    " '0.2b': 2.849509312,\n",
    " '0.5b': 6.39294768,\n",
    " '1b': 13.301467744}\n",
    "\n",
    "mixtral_32x_topk_4_activeparam_config = {'15m': 0.35767904,\n",
    " '44m': 1.07293872,\n",
    " '0.1b': 2.38419904,\n",
    " '0.2b': 5.640836032,\n",
    " '0.5b': 12.6553784,\n",
    " '1b': 26.331401824}\n",
    "\n",
    "mixtral_64x_topk_8_activeparam_config = {'15m': 0.7116656,\n",
    " '44m': 2.13480624,\n",
    " '0.1b': 4.74380224,\n",
    " '0.2b': 11.223489472,\n",
    " '0.5b': 25.18023984,\n",
    " '1b': 52.391269984}\n",
    "\n",
    "token_multiplier_config = {'15m': 128 * 1024 / 1e9,\n",
    " '44m': 256* 1024/ 1e9,\n",
    " '0.1b': 512* 1024/ 1e9,\n",
    " '0.2b': 512* 1024/ 1e9,\n",
    " '0.5b': 512* 1024/ 1e9,\n",
    " '1b': 1024* 1024/ 1e9} "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f06ffdbf",
   "metadata": {},
   "source": [
    "# load data (base2code)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "91930e5b",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "import numpy as np\n",
    "\n",
    "with open(\"../data_post_icml/base2code.txt\", \"r\") as file:\n",
    "    lines = file.readlines()\n",
    "data = []\n",
    "for line in lines:\n",
    "    parts = line.split()\n",
    "    # print(parts)\n",
    "    model_key = parts[0]  # Extract the key part like '15m', '44m', etc.\n",
    "    if model_key in mixtral_model_config:\n",
    "        data.append({\n",
    "            \"Model\": parts[0],\n",
    "            \"N_dense\": llama_model_config[model_key],\n",
    "            \"N_active\": mixtral_model_activeparam_config[model_key],\n",
    "            \"N_totalactive\": mixtral_total_model_activeparam_config[model_key],\n",
    "            \"D1\": int(parts[2]) * token_multiplier_config[model_key] ,\n",
    "            \"D2\": int(parts[3]) * token_multiplier_config[model_key],\n",
    "            # \"Loss\": float(parts[4].replace('E+00', ''))\n",
    "            \"Loss\": float(parts[4])\n",
    "        })\n",
    "upcycle_8x=pd.DataFrame(data)\n",
    "\n",
    "upcycle_8x = upcycle_8x.dropna()\n",
    "x2 = upcycle_8x['D2'].values  * 1e9+1\n",
    "x1 = upcycle_8x['D1'].values  * 1e9+1\n",
    "# x1 = upcycle_8x['Loss_dense'].values / (1e9+1)\n",
    "x3 = upcycle_8x['N_dense'].values  *  1e9\n",
    "y = upcycle_8x['Loss'].values \n",
    "\n",
    "inp_upc = np.column_stack([x1, x2,y, x3])\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "59a5d949",
   "metadata": {},
   "source": [
    "## fit "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 90,
   "id": "d69baf45",
   "metadata": {},
   "outputs": [],
   "source": [
    "import autograd.numpy as np\n",
    "from autograd import grad\n",
    "from scipy.optimize import minimize\n",
    "from tqdm import tqdm\n",
    "\n",
    "def custom_logsumexp(terms):\n",
    "    max_term = np.max(terms, axis=0)\n",
    "    sum_exp = np.sum(np.exp(terms - max_term), axis=0)\n",
    "    return max_term + np.log(sum_exp)\n",
    "\n",
    "\n",
    "def loss_scipy_upcycle(params, inp, loss_type=\"huber\", delta = 1e-3):\n",
    "    # a, b, d1 = params\n",
    "    a, b, c, e, d1, d2, d3 = params\n",
    "    x1, x2, x3, y = inp[:, 0], inp[:, 1], inp[:, 3], inp[:, 2]\n",
    "\n",
    "    term1 = d1  - a * np.log( x1) - b * np.log(x2) + c * np.log(x1)* np.log(x2)\n",
    "    term2 = d2  - e * np.log( x3) \n",
    "    term3 = d3 * np.ones_like(x1) # use fit from scratch models\n",
    "\n",
    "    # Use custom_logsumexp for numerical stability\n",
    "    terms = np.stack([term1,term2, term3], axis=0)\n",
    "    post_lse = custom_logsumexp(terms)\n",
    "    \n",
    "    if loss_type == \"huber\":\n",
    "        residual = post_lse - np.log(y)\n",
    "        loss = np.where(np.abs(residual) <= delta,\n",
    "                                0.5 * residual**2,\n",
    "                                delta * (np.abs(residual) - 0.5 * delta)).sum()\n",
    "    elif loss_type == \"msle\":\n",
    "        loss = ((post_lse - np.log(y))**2).sum()\n",
    "    elif loss_type == \"mse\":\n",
    "        loss = ((np.exp(post_lse) - y)**2).sum()\n",
    "    else:\n",
    "        raise NotImplementedError(f\"loss {loss_type} not implemented!\")\n",
    "    \n",
    "    return loss \n",
    "\n",
    "def objective_fn(params, inp):\n",
    "    loss3 = loss_scipy_upcycle(params, inp)\n",
    "    total_loss = loss3\n",
    "    return total_loss\n",
    "\n",
    "def random_search_optimize( inp, n_iter=100, early_stopping_rounds=10):\n",
    "    min_loss = float('inf')\n",
    "    best_params = None\n",
    "    no_improve_rounds = 0\n",
    "\n",
    "    for _ in range(n_iter):\n",
    "        # Sample random parameters from the specified ranges\n",
    "        a = np.random.uniform(0, 1)\n",
    "        b = np.random.uniform(0, 1)\n",
    "        c = np.random.uniform(0, 1)\n",
    "        e = np.random.uniform(0, 1)\n",
    "        d1 = np.random.uniform(0, 10)\n",
    "        d2 = np.random.uniform(0, 10)\n",
    "        d3 = np.random.uniform(0, 10)\n",
    "        init_params = (a, b,c, e, d1, d2, d3) # For upcycle model\n",
    "        \n",
    "        objective_grad = grad(objective_fn)\n",
    "        \n",
    "        # Run optimization with the Jacobian\n",
    "        result = minimize(\n",
    "            objective_fn, \n",
    "            init_params, \n",
    "            args=inp,\n",
    "            method='BFGS', \n",
    "            # method='L-BFGS-B', \n",
    "            jac=objective_grad\n",
    "        )\n",
    "        \n",
    "        l = result.fun\n",
    "        params = result.x\n",
    "        \n",
    "        # Update best params if loss is lower\n",
    "        if l < min_loss:\n",
    "            min_loss = l\n",
    "            best_params = params\n",
    "            no_improve_rounds = 0  # Reset early stopping counter\n",
    "        else:\n",
    "            no_improve_rounds += 1\n",
    "\n",
    "        # Check for early stopping\n",
    "        if no_improve_rounds >= early_stopping_rounds:\n",
    "            print(f\"Early stopping after {early_stopping_rounds} rounds with no improvement.\")\n",
    "            break\n",
    "\n",
    "    return min_loss, best_params\n",
    "\n",
    "from itertools import product\n",
    "import random\n",
    "\n",
    "def grid_search_optimize(inp, param_grid, early_stopping_rounds=10):\n",
    "    min_loss = float('inf')\n",
    "    best_params = None\n",
    "    no_improve_rounds = 0\n",
    "\n",
    "    # Generate all combinations of parameter values\n",
    "    param_combinations = list(product(\n",
    "        param_grid['a'], \n",
    "        param_grid['b'], \n",
    "        param_grid['c'], \n",
    "        param_grid['e'], \n",
    "        param_grid['d1'], \n",
    "        param_grid['d2'], \n",
    "        param_grid['d3'], \n",
    "    ))\n",
    "    random.shuffle(param_combinations)\n",
    "\n",
    "    for params in param_combinations:\n",
    "        a, b, c, e, d1, d2, d3 = params\n",
    "        init_params = (a, b, c, e, d1, d2, d3)  # For upcycle model\n",
    "\n",
    "        objective_grad = grad(objective_fn)\n",
    "\n",
    "        # Run optimization with the Jacobian\n",
    "        result = minimize(\n",
    "            objective_fn, \n",
    "            init_params, \n",
    "            args=inp,\n",
    "            method='BFGS', \n",
    "            jac=objective_grad\n",
    "        )\n",
    "\n",
    "        l = result.fun\n",
    "        optimized_params = result.x\n",
    "\n",
    "        # Update best params if loss is lower\n",
    "        if l < min_loss:\n",
    "            min_loss = l\n",
    "            best_params = optimized_params\n",
    "            no_improve_rounds = 0  # Reset early stopping counter\n",
    "        else:\n",
    "            no_improve_rounds += 1\n",
    "\n",
    "        # Check for early stopping\n",
    "        if no_improve_rounds >= early_stopping_rounds:\n",
    "            print(f\"Early stopping after {early_stopping_rounds} rounds with no improvement.\")\n",
    "            break\n",
    "\n",
    "    return min_loss, best_params\n",
    "\n",
    "from scipy.special import logsumexp\n",
    "\n",
    "def evaluate_goodness_of_fit(params, inp):\n",
    "    # Extract relevant parameters for each model\n",
    "   \n",
    "    a, b, c, e, d1, d2, d3 = params\n",
    "    x1, x2, x3, y_actual_upcycle = inp[:, 0], inp[:, 1], inp[:, 3], inp[:, 2]\n",
    "\n",
    "    term1 = d1  - a * np.log( x1) - b * np.log(x2) + c * np.log(x1)* np.log(x2)\n",
    "    term2 = d2  - e * np.log( x3) \n",
    "    term3 = d3 * np.ones_like(x1) # use fit from scratch models\n",
    "\n",
    "    # Use custom_logsumexp for numerical stability\n",
    "    terms = np.stack([term1,term2, term3], axis=0)\n",
    "    post_lse = custom_logsumexp(terms)\n",
    "    \n",
    "    # 3. Evaluate Upcycle Model\n",
    "    y_pred_upcycle = np.exp(post_lse)\n",
    "\n",
    "    rms_error_upcycle = np.sqrt(np.mean((y_actual_upcycle - y_pred_upcycle) ** 2))\n",
    "    rmsle_error_upcycle = np.sqrt(np.mean((np.log1p(y_actual_upcycle) - np.log1p(y_pred_upcycle)) ** 2))\n",
    "    ss_total_upcycle = np.sum((y_actual_upcycle - np.mean(y_actual_upcycle)) ** 2)\n",
    "    ss_residual_upcycle = np.sum((y_actual_upcycle - y_pred_upcycle) ** 2)\n",
    "    r2_error_upcycle = 1 - (ss_residual_upcycle / ss_total_upcycle)\n",
    "\n",
    "    # y_actual.append(y_actual_upcycle)\n",
    "    # y_predict.append(y_pred_upcycle)\n",
    "\n",
    "    results = {}\n",
    "    results['upcycle'] = {\n",
    "        'rms_error': rms_error_upcycle,\n",
    "        'rmsle_error': rmsle_error_upcycle,\n",
    "        'r2_error': r2_error_upcycle\n",
    "    }\n",
    "\n",
    "    # Output results\n",
    "    for model, metrics in results.items():\n",
    "        print(f\"\\nModel: {model.capitalize()}\")\n",
    "        print(f\"RMS error: {metrics['rms_error']:.3f}\")\n",
    "        print(f\"RMSLE error: {metrics['rmsle_error']:.3f}\")\n",
    "        print(f\"R² error: {metrics['r2_error']:.3f}\")\n",
    "\n",
    "    return results, y_actual_upcycle, y_pred_upcycle"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 81,
   "id": "63d51368",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Early stopping after 400 rounds with no improvement.\n",
      "0.0008864756269784875 [-2.62897236e-02  7.13682121e-02 -2.09894959e-03  2.75669796e-01\n",
      "  1.37596335e+00  3.87024245e+00 -1.24059491e+00]\n",
      "\n",
      "Model: Upcycle\n",
      "RMS error: 0.014\n",
      "RMSLE error: 0.007\n",
      "R² error: 0.991\n",
      "{'upcycle': {'rms_error': np.float64(0.014219815413633579), 'rmsle_error': np.float64(0.007127045203522519), 'r2_error': np.float64(0.9907843116670324)}}\n"
     ]
    }
   ],
   "source": [
    "# loss, param = random_search_optimize(inp_upc)\n",
    "# sorted_array = inp_upc[inp_upc[:, 2].argsort()]\n",
    "# split_index = int(len(inp_upc) * 0)\n",
    "sorted_array = inp_upc[inp_upc[:, 2].argsort()]\n",
    "split_index = int(len(inp_upc) * 0.1)\n",
    "\n",
    "\n",
    "param_grid = {\n",
    "    'a': np.linspace(0, 1, 5),  # 5 equally spaced values between 0 and 1\n",
    "    'b': np.linspace(0, 1, 5),\n",
    "    'c': np.linspace(0, 1, 5),\n",
    "    'e': np.linspace(0, 1, 5),\n",
    "    'd1': np.linspace(-5, 5, 5),\n",
    "    'd2': np.linspace(-5, 5, 5),\n",
    "    'd3': np.linspace(-5, 5, 5),\n",
    "}\n",
    "\n",
    "# Split the array\n",
    "split_80 = sorted_array[split_index:]  # 80% with higher values in the 4th column\n",
    "split_20 = sorted_array[:split_index]  # 20% with lowest values in the 4th column\n",
    "loss_train, params_train = grid_search_optimize(split_80, param_grid=param_grid, early_stopping_rounds=400)\n",
    "print(loss_train, params_train)\n",
    "\n",
    "\n",
    "# loss, param = random_search_optimize(inp_upc)\n",
    "results_train = evaluate_goodness_of_fit(params_train, split_80)\n",
    "print(results_train[0])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 83,
   "id": "86ab4536",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Model: Upcycle\n",
      "RMS error: 0.014\n",
      "RMSLE error: 0.007\n",
      "R² error: 0.991\n",
      "\n",
      "Model: Upcycle\n",
      "RMS error: 0.008\n",
      "RMSLE error: 0.004\n",
      "R² error: 0.722\n",
      "{'upcycle': {'rms_error': np.float64(0.014219815413633579), 'rmsle_error': np.float64(0.007127045203522519), 'r2_error': np.float64(0.9907843116670324)}}\n",
      "{'upcycle': {'rms_error': np.float64(0.007502933307505452), 'rmsle_error': np.float64(0.0041244585591124995), 'r2_error': np.float64(0.7222990117438777)}}\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# loss, param = random_search_optimize(inp_upc)\n",
    "results_train = evaluate_goodness_of_fit(params_train, split_80)\n",
    "results_val = evaluate_goodness_of_fit(params_train, split_20)\n",
    "print(results_train[0])\n",
    "print(results_val[0])\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "plt.scatter(results_train[1], results_train[2],label='Fitted')\n",
    "plt.scatter(results_val[1], results_val[2],label=\"Validated\")\n",
    "# plt.grid()\n",
    "plt.xlabel(r'Actual Loss',fontsize=20)\n",
    "plt.ylabel(r'Predicted Loss',fontsize=20)\n",
    "plt.legend(loc=\"best\", prop={'size': 14}, title_fontsize=16)\n",
    "# plt.plot([2, 3.35],[2, 3.35], color='red', linestyle='--', label='Ideal Fit')\n",
    "# plt.show()\n",
    "plt.plot([0.7, 1.5],[0.7, 1.5], color='red', linestyle='--', label='Ideal Fit')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3b208f6f",
   "metadata": {},
   "source": [
    "## fit full"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 91,
   "id": "207f1597",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Early stopping after 400 rounds with no improvement.\n",
      "0.0009464815209091127 [ 4.82977689e-02  1.25594159e-01  1.48598516e-03  2.37568149e-01\n",
      "  2.71219738e+00  3.30448377e+00 -2.25638122e+00]\n",
      "\n",
      "Model: Upcycle\n",
      "RMS error: 0.014\n",
      "RMSLE error: 0.007\n",
      "R² error: 0.992\n",
      "{'upcycle': {'rms_error': np.float64(0.013866037560909775), 'rmsle_error': np.float64(0.006946114578464093), 'r2_error': np.float64(0.9922710321016467)}}\n"
     ]
    }
   ],
   "source": [
    "# loss, param = random_search_optimize(inp_upc)\n",
    "# sorted_array = inp_upc[inp_upc[:, 2].argsort()]\n",
    "# split_index = int(len(inp_upc) * 0)\n",
    "sorted_array = inp_upc[inp_upc[:, 2].argsort()]\n",
    "split_index = int(len(inp_upc) * 0.)\n",
    "\n",
    "\n",
    "param_grid = {\n",
    "    'a': np.linspace(0, 1, 5),  # 5 equally spaced values between 0 and 1\n",
    "    'b': np.linspace(0, 1, 5),\n",
    "    'c': np.linspace(0, 1, 5),\n",
    "    'e': np.linspace(0, 1, 5),\n",
    "    'd1': np.linspace(-5, 5, 5),\n",
    "    'd2': np.linspace(-5, 5, 5),\n",
    "    'd3': np.linspace(-5, 5, 5),\n",
    "}\n",
    "\n",
    "# Split the array\n",
    "split_80 = sorted_array[split_index:]  # 80% with higher values in the 4th column\n",
    "# split_20 = sorted_array[:split_index]  # 20% with lowest values in the 4th column\n",
    "loss_train, params_train = grid_search_optimize(split_80, param_grid=param_grid, early_stopping_rounds=400)\n",
    "print(loss_train, params_train)\n",
    "\n",
    "\n",
    "# loss, param = random_search_optimize(inp_upc)\n",
    "results_train = evaluate_goodness_of_fit(params_train, split_80)\n",
    "print(results_train[0])\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "222d8b34",
   "metadata": {},
   "source": [
    "## fit full (fix E)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 141,
   "id": "5be55ea8",
   "metadata": {},
   "outputs": [],
   "source": [
    "import autograd.numpy as np\n",
    "from autograd import grad\n",
    "from scipy.optimize import minimize\n",
    "from tqdm import tqdm\n",
    "\n",
    "def custom_logsumexp(terms):\n",
    "    max_term = np.max(terms, axis=0)\n",
    "    sum_exp = np.sum(np.exp(terms - max_term), axis=0)\n",
    "    return max_term + np.log(sum_exp)\n",
    "\n",
    "\n",
    "def loss_scipy_upcycle(params, inp, loss_type=\"huber\", delta = 1e-3):\n",
    "    # a, b, d1 = params\n",
    "    a, b, c, e, d1, d2 = params\n",
    "    x1, x2, x3, y = inp[:, 0], inp[:, 1], inp[:, 3], inp[:, 2]\n",
    "    d3 = -1.23422397\n",
    "\n",
    "    term1 = d1  - a * np.log( x1) - b * np.log(x2) + c * np.log(x1)* np.log(x2)\n",
    "    term2 = d2  - e * np.log( x3) \n",
    "    term3 = d3 * np.ones_like(x1) # use fit from scratch models\n",
    "\n",
    "    # Use custom_logsumexp for numerical stability\n",
    "    terms = np.stack([term1,term2, term3], axis=0)\n",
    "    post_lse = custom_logsumexp(terms)\n",
    "    \n",
    "    if loss_type == \"huber\":\n",
    "        residual = post_lse - np.log(y)\n",
    "        loss = np.where(np.abs(residual) <= delta,\n",
    "                                0.5 * residual**2,\n",
    "                                delta * (np.abs(residual) - 0.5 * delta)).sum()\n",
    "    elif loss_type == \"msle\":\n",
    "        loss = ((post_lse - np.log(y))**2).sum()\n",
    "    elif loss_type == \"mse\":\n",
    "        loss = ((np.exp(post_lse) - y)**2).sum()\n",
    "    else:\n",
    "        raise NotImplementedError(f\"loss {loss_type} not implemented!\")\n",
    "    \n",
    "    return loss \n",
    "\n",
    "def objective_fn(params, inp):\n",
    "    loss3 = loss_scipy_upcycle(params, inp)\n",
    "    total_loss = loss3\n",
    "    return total_loss\n",
    "\n",
    "def random_search_optimize( inp, n_iter=100, early_stopping_rounds=10):\n",
    "    min_loss = float('inf')\n",
    "    best_params = None\n",
    "    no_improve_rounds = 0\n",
    "\n",
    "    for _ in range(n_iter):\n",
    "        # Sample random parameters from the specified ranges\n",
    "        a = np.random.uniform(0, 1)\n",
    "        b = np.random.uniform(0, 1)\n",
    "        c = np.random.uniform(0, 1)\n",
    "        e = np.random.uniform(0, 1)\n",
    "        d1 = np.random.uniform(0, 10)\n",
    "        d2 = np.random.uniform(0, 10)\n",
    "        # d3 = np.random.uniform(0, 10)\n",
    "        init_params = (a, b,c, e, d1, d2) # For upcycle model\n",
    "        \n",
    "        objective_grad = grad(objective_fn)\n",
    "        \n",
    "        # Run optimization with the Jacobian\n",
    "        result = minimize(\n",
    "            objective_fn, \n",
    "            init_params, \n",
    "            args=inp,\n",
    "            method='BFGS', \n",
    "            # method='L-BFGS-B', \n",
    "            jac=objective_grad\n",
    "        )\n",
    "        \n",
    "        l = result.fun\n",
    "        params = result.x\n",
    "        \n",
    "        # Update best params if loss is lower\n",
    "        if l < min_loss:\n",
    "            min_loss = l\n",
    "            best_params = params\n",
    "            no_improve_rounds = 0  # Reset early stopping counter\n",
    "        else:\n",
    "            no_improve_rounds += 1\n",
    "\n",
    "        # Check for early stopping\n",
    "        if no_improve_rounds >= early_stopping_rounds:\n",
    "            print(f\"Early stopping after {early_stopping_rounds} rounds with no improvement.\")\n",
    "            break\n",
    "\n",
    "    return min_loss, best_params\n",
    "\n",
    "from itertools import product\n",
    "import random\n",
    "\n",
    "def grid_search_optimize(inp, param_grid, early_stopping_rounds=10):\n",
    "    min_loss = float('inf')\n",
    "    best_params = None\n",
    "    no_improve_rounds = 0\n",
    "\n",
    "    # Generate all combinations of parameter values\n",
    "    param_combinations = list(product(\n",
    "        param_grid['a'], \n",
    "        param_grid['b'], \n",
    "        param_grid['c'], \n",
    "        param_grid['e'], \n",
    "        param_grid['d1'], \n",
    "        param_grid['d2'], \n",
    "        # param_grid['d3'], \n",
    "    ))\n",
    "    random.shuffle(param_combinations)\n",
    "\n",
    "    for params in param_combinations:\n",
    "        a, b, c, e, d1, d2 = params\n",
    "        init_params = (a, b, c, e, d1, d2)  # For upcycle model\n",
    "\n",
    "        objective_grad = grad(objective_fn)\n",
    "\n",
    "        # Run optimization with the Jacobian\n",
    "        result = minimize(\n",
    "            objective_fn, \n",
    "            init_params, \n",
    "            args=inp,\n",
    "            method='BFGS', \n",
    "            jac=objective_grad\n",
    "        )\n",
    "\n",
    "        l = result.fun\n",
    "        optimized_params = result.x\n",
    "\n",
    "        # Update best params if loss is lower\n",
    "        if l < min_loss:\n",
    "            min_loss = l\n",
    "            best_params = optimized_params\n",
    "            no_improve_rounds = 0  # Reset early stopping counter\n",
    "        else:\n",
    "            no_improve_rounds += 1\n",
    "\n",
    "        # Check for early stopping\n",
    "        if no_improve_rounds >= early_stopping_rounds:\n",
    "            print(f\"Early stopping after {early_stopping_rounds} rounds with no improvement.\")\n",
    "            break\n",
    "\n",
    "    return min_loss, best_params\n",
    "\n",
    "from scipy.special import logsumexp\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 142,
   "id": "8a19b028",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Early stopping after 400 rounds with no improvement.\n",
      "0.000960376618283331 [-1.48158388e-02  8.93384352e-02 -1.55871204e-03  2.60839790e-01\n",
      "  1.72662301e+00  3.65180680e+00]\n"
     ]
    }
   ],
   "source": [
    "# loss, param = random_search_optimize(inp_upc)\n",
    "# sorted_array = inp_upc[inp_upc[:, 2].argsort()]\n",
    "# split_index = int(len(inp_upc) * 0)\n",
    "\n",
    "sorted_array = inp_upc[inp_upc[:, 2].argsort()]\n",
    "split_index = int(len(inp_upc) * 0.)\n",
    "\n",
    "\n",
    "param_grid = {\n",
    "    'a': np.linspace(0, 1, 5),  # 5 equally spaced values between 0 and 1\n",
    "    'b': np.linspace(0, 1, 5),\n",
    "    'c': np.linspace(0, 1, 5),\n",
    "    'e': np.linspace(0, 1, 5),\n",
    "    'd1': np.linspace(-5, 5, 5),\n",
    "    'd2': np.linspace(-5, 5, 5),\n",
    "    # 'd3': np.linspace(-5, 5, 5),\n",
    "}\n",
    "\n",
    "# Split the array\n",
    "split_80 = sorted_array[split_index:]  # 80% with higher values in the 4th column\n",
    "# split_20 = sorted_array[:split_index]  # 20% with lowest values in the 4th column\n",
    "loss_train, params_train = grid_search_optimize(split_80, param_grid=param_grid, early_stopping_rounds=400)\n",
    "print(loss_train, params_train)\n",
    "\n",
    "\n",
    "# loss, param = random_search_optimize(inp_upc)\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": ".env",
   "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.13.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
