{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "based on https://pennylane.ai/qml/demos/tutorial_expressivity_fourier_series/ and the paper *The effect of data encoding on\n",
    "the expressive power of variational quantum machine learning models* by [Schuld, Sweke, and Meyer (2020)](https://arxiv.org/abs/2008.08605)"
   ]
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T10:39:10.396235Z",
     "start_time": "2025-09-16T10:39:06.866353Z"
    }
   },
   "cell_type": "code",
   "source": [
    "from pathlib import Path\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "\n",
    "import pennylane as qml\n",
    "import jax, jaxlib\n",
    "import jax.numpy as jnp\n",
    "import optax\n",
    "\n",
    "from sklearn.metrics import r2_score\n",
    "import pandas as pd\n",
    "import pickle\n"
   ],
   "outputs": [],
   "execution_count": 1
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T10:39:10.404937Z",
     "start_time": "2025-09-16T10:39:10.398340Z"
    }
   },
   "cell_type": "code",
   "source": [
    "seed = 42\n",
    "np.random.seed(seed)"
   ],
   "outputs": [],
   "execution_count": 2
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### setting up the building blocks of the model"
   ]
  },
  {
   "cell_type": "code",
   "source": [
    "# define Feature Map S\n",
    "# contains prefactor alpha to allow for exponentially increasing frequencies\n",
    "\n",
    "def S(alpha, x, num_wires):\n",
    "    \"\"\"Feature Map with pre-factor\"\"\"\n",
    "    for w in range(num_wires):\n",
    "        qml.RX(alpha[w]*x, wires=w)"
   ],
   "metadata": {
    "collapsed": false,
    "ExecuteTime": {
     "end_time": "2025-09-16T10:39:10.405155Z",
     "start_time": "2025-09-16T10:39:10.402044Z"
    }
   },
   "outputs": [],
   "execution_count": 3
  },
  {
   "cell_type": "code",
   "source": [
    "# define basic circuit architecture for Ansatz W - see main Opt Loop for a circuit drawing to check optically\n",
    "\n",
    "def W_ladder(theta, trainable_block_layers, num_wires):\n",
    "    \"\"\"Ansatz layer with variable number of train blocks consisting of CNOTs and Rot gates\"\"\"\n",
    "    for i in range(trainable_block_layers):\n",
    "        qml.Rot(theta[i][num_wires-1][0], theta[i][num_wires-1][1], theta[i][num_wires-1][2], wires = 0)\n",
    "        for j in range(num_wires-1):\n",
    "            qml.CNOT(wires = [j, j+1])\n",
    "            qml.Rot(theta[i][j][0], theta[i][j][1], theta[i][j][2], wires = j+1)\n",
    "        "
   ],
   "metadata": {
    "collapsed": false,
    "ExecuteTime": {
     "end_time": "2025-09-16T10:39:10.413745Z",
     "start_time": "2025-09-16T10:39:10.407525Z"
    }
   },
   "outputs": [],
   "execution_count": 4
  },
  {
   "cell_type": "code",
   "source": [
    "# weights initialisation\n",
    "\n",
    "def random_weights_uniform(num_serial_ansatz_layers, trainable_block_layers, num_wires, num_rot_params):\n",
    "    return 2 * np.pi * np.random.random(size=(num_serial_ansatz_layers, trainable_block_layers, num_wires, num_rot_params))\n"
   ],
   "metadata": {
    "collapsed": false,
    "ExecuteTime": {
     "end_time": "2025-09-16T10:39:10.422912Z",
     "start_time": "2025-09-16T10:39:10.412897Z"
    }
   },
   "outputs": [],
   "execution_count": 5
  },
  {
   "cell_type": "code",
   "source": [
    "# enable jax for faster execution\n",
    "\n",
    "jax.config.update(\"jax_enable_x64\", True)"
   ],
   "metadata": {
    "collapsed": false,
    "ExecuteTime": {
     "end_time": "2025-09-16T10:39:10.431491Z",
     "start_time": "2025-09-16T10:39:10.415318Z"
    }
   },
   "outputs": [],
   "execution_count": 6
  },
  {
   "cell_type": "code",
   "source": [
    "# define MSE as loss function\n",
    "\n",
    "def square_loss(targets, predictions):\n",
    "    loss = 0\n",
    "    for t, p in zip(targets, predictions):\n",
    "        loss += (t - p) ** 2\n",
    "    loss = loss / len(targets)\n",
    "    return 0.5 * loss"
   ],
   "metadata": {
    "collapsed": false,
    "ExecuteTime": {
     "end_time": "2025-09-16T10:39:10.431688Z",
     "start_time": "2025-09-16T10:39:10.420218Z"
    }
   },
   "outputs": [],
   "execution_count": 7
  },
  {
   "cell_type": "code",
   "source": [
    "# update parameters with jax / jit\n",
    "\n",
    "@jax.jit\n",
    "def update(params, opt_state, x, y):\n",
    "    loss, grads = jax.value_and_grad(cost)(params, x, y)\n",
    "    updates, opt_state = opt.update(grads, opt_state, params)\n",
    "    params = optax.apply_updates(params, updates)\n",
    "    return params, opt_state, loss, grads\n"
   ],
   "metadata": {
    "collapsed": false,
    "ExecuteTime": {
     "end_time": "2025-09-16T10:39:10.432293Z",
     "start_time": "2025-09-16T10:39:10.424947Z"
    }
   },
   "outputs": [],
   "execution_count": 8
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T10:39:10.432393Z",
     "start_time": "2025-09-16T10:39:10.427735Z"
    }
   },
   "cell_type": "code",
   "source": [
    "# select initial prefactors alpha for exponential encoding, add separate inner list [] for each serial FM layer\n",
    "\n",
    "initial_alpha = [[1.0, 3.0, 9.0]]"
   ],
   "outputs": [],
   "execution_count": 9
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T10:39:10.451818Z",
     "start_time": "2025-09-16T10:39:10.431401Z"
    }
   },
   "cell_type": "code",
   "source": [
    "# load target functions dataset with 10 different coefficient sets\n",
    "\n",
    "with open('../data/input/1d_datasets.pkl', 'rb') as f:\n",
    "    loaded_datasets = pickle.load(f)\n"
   ],
   "outputs": [],
   "execution_count": 10
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "outputs": [
    {
     "data": {
      "text/plain": "dict_keys(['x_all', 'y_all', 'x_train', 'x_test', 'y_train', 'y_test'])"
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# quick check to confirm that dataset loading has worked\n",
    "\n",
    "loaded_datasets[0].keys()"
   ],
   "metadata": {
    "collapsed": false,
    "ExecuteTime": {
     "end_time": "2025-09-16T10:39:10.459347Z",
     "start_time": "2025-09-16T10:39:10.452157Z"
    }
   }
  },
  {
   "cell_type": "code",
   "metadata": {
    "collapsed": false,
    "ExecuteTime": {
     "end_time": "2025-09-16T10:40:28.097483Z",
     "start_time": "2025-09-16T10:39:10.454853Z"
    }
   },
   "source": [
    "# Optimization Loop\n",
    "\n",
    "# use this hyperparameter to loop over increasing number of parameters\n",
    "max_trainable_block_layers = 11 # max number of train blocks (excl) within one ansatz layer between FMs\n",
    "\n",
    "num_runs_per_function = 10\n",
    "max_steps = 5000\n",
    "lr = 0.001\n",
    "batch_size = 40\n",
    "num_wires = 3\n",
    "num_serial_ansatz_layers = 2 # one extra Ansatz layer in front, so 2 ansatz layers for 1 serial FM\n",
    "num_rot_params = 3 # number of rotational parameters within 1 train Block\n",
    "weights_scaling = 1 # initial weights scaling factor (use for near zero weights initialization)\n",
    "\n",
    "# Initialize an empty DataFrame to store metrics\n",
    "metrics_df = pd.DataFrame(columns=[\"Approach\", \"Degree\", \"Run\", \"R2_Score\", \"numParams\"])\n",
    "\n",
    "# define path to save results\n",
    "model_name = f\"denseFreqs_parallel_ternary\"\n",
    "fileDir = Path(f'../results/')    \n",
    "fileDir.mkdir(parents=True, exist_ok=True)\n",
    "filename = Path(f\"{model_name}_{int(num_wires)}wires_{num_serial_ansatz_layers-1}ser_targetMultipleFourier1d\")\n",
    "filepathDataframe = Path.joinpath(fileDir, filename).with_suffix('.csv')\n",
    "\n",
    "\n",
    "for trainable_block_layers in range(1, max_trainable_block_layers):\n",
    "    \n",
    "    print(f\"running {trainable_block_layers} train blocks of {max_trainable_block_layers}\")\n",
    "\n",
    "    dev = qml.device(\"default.qubit\", wires = num_wires)\n",
    "    # define model within loop to allow for additional train blocks / parameters in circuit\n",
    "    @qml.qnode(dev)\n",
    "    def standard_parallel_quantum_model(alpha, weights, x=None):\n",
    "        W_ladder(weights[0], trainable_block_layers, num_wires)\n",
    "        for i in range(num_serial_ansatz_layers-1):\n",
    "            S(alpha[i], x, num_wires)\n",
    "            W_ladder(weights[i+1], trainable_block_layers, num_wires)\n",
    "\n",
    "        return qml.expval(qml.PauliZ(wires=num_wires-1))\n",
    "\n",
    "    # get circuit plotted for inspection\n",
    "    fig, ax = qml.draw_mpl(standard_parallel_quantum_model, expansion_strategy=\"device\")(initial_alpha, random_weights_uniform(num_serial_ansatz_layers, trainable_block_layers, num_wires, num_rot_params), 0.5)\n",
    "    plt.show()\n",
    "\n",
    "    # transform circuit into jit\n",
    "    standard_parallel_quantum_model_jit = jax.jit(standard_parallel_quantum_model)\n",
    "    alpha = jnp.array(initial_alpha)\n",
    "    \n",
    "    # define cost function\n",
    "    def cost(params, x, y):\n",
    "        weights = params[\"weights\"]\n",
    "        predictions = [standard_parallel_quantum_model_jit(alpha, weights, x_) for x_ in x]\n",
    "        return square_loss(y, predictions).squeeze()\n",
    "    \n",
    "    # run circuit for full set of target functions\n",
    "    for func in range(len(loaded_datasets)):\n",
    "    \n",
    "        print(f\"running function {func+1} of {len(loaded_datasets)}\")\n",
    "        \n",
    "        # extract data from loaded datasets for specific target function\n",
    "        dataset = loaded_datasets[func]\n",
    "        x_all = dataset[\"x_all\"]\n",
    "        x_train = dataset[\"x_train\"]\n",
    "        x_test = dataset[\"x_test\"]\n",
    "        y_all = dataset[\"y_all\"]\n",
    "        y_train = dataset[\"y_train\"]\n",
    "        y_test = dataset[\"y_test\"]\n",
    "\n",
    "        # run for several runs to avoid any seed luck with the initialization of the weights\n",
    "        for run in range(num_runs_per_function):\n",
    "            \n",
    "            # create new randome weights and transform into jnp array\n",
    "            weights_rand = random_weights_uniform(num_serial_ansatz_layers, trainable_block_layers, num_wires, num_rot_params)\n",
    "            weights_flex = weights_scaling * weights_rand # scale initial weights (for near zero weights initialization)\n",
    "            weights_flex = jnp.array(weights_flex)\n",
    "            \n",
    "            # define optimizer\n",
    "            opt = optax.adam(lr)\n",
    "    \n",
    "            # set up params dictionary to be used in update function\n",
    "            params = {'weights': weights_flex}\n",
    "            opt_state = opt.init(params)\n",
    "            cst = [cost(params, x_train, y_train)]  # initial cost\n",
    "            \n",
    "            # run optimization\n",
    "            for step in range(max_steps):\n",
    "    \n",
    "                # select batch of data\n",
    "                batch_index = np.random.randint(0, len(x_train), (batch_size,))\n",
    "                x_batch = jnp.array(x_train[batch_index])\n",
    "                y_batch = jnp.array(y_train[batch_index])\n",
    "    \n",
    "                # update parameters\n",
    "                params, opt_state, c, gradients = update(params, opt_state, x_batch, y_batch)\n",
    "                \n",
    "                # update cost history and print periodically\n",
    "                cst.append(c)\n",
    "                print_freq = 1000\n",
    "                if (step + 1) % print_freq == 0:\n",
    "                    print(\"Cost at step {0:3}: {1}\".format(step + 1, c))\n",
    "    \n",
    "            # calculate R2 on test set with trained weights\n",
    "            weights = params[\"weights\"]\n",
    "            pred_test = [standard_parallel_quantum_model_jit(alpha, weights, x_) for x_ in x_test]  # test set for R2 calc\n",
    "            r2 = r2_score(y_test, pred_test)\n",
    "            numParams = jnp.size(weights)\n",
    "            \n",
    "            print(\"numParams: \", numParams) # for periodic output as status update\n",
    "            \n",
    "            # Append metrics to the DataFrame\n",
    "            metrics_df = pd.concat([metrics_df, pd.DataFrame({\n",
    "                \"Approach\": f\"{model_name}_{func}\",\n",
    "                \"Run\": [run],\n",
    "                \"Cost\": [c],\n",
    "                \"R2_Score\": [r2],\n",
    "                \"numParams\": [numParams]\n",
    "            })], ignore_index=True)\n",
    "    \n",
    "            # Save to a file after each iteration\n",
    "            metrics_df.to_csv(filepathDataframe, index=False)\n",
    "    \n",
    "            # print metrics for interim updates\n",
    "            print(f\"Approach {model_name}: Run: {run}, Cost = {c:.4f}, R2_Score = {r2:.4f}\")\n",
    "    "
   ],
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "running 1 train blocks of 11\n"
     ]
    },
    {
     "data": {
      "text/plain": "<Figure size 1200x400 with 1 Axes>",
      "image/png": "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BTpw4IBOnz7tvotjRESEWrVqpc6dO6tz586y2+166qmn6t3Gnj17NG7cOI+7TNIQ8z8aYgAAGODrr7/Wtm3bFBISopEjRxodTpMbNWqUsrKytHXrVn3zzTfq1KmT0SEBQFCgfgD+dfr0aeXm5mrbtm3au3evvv76a50/f/6ivxMaGqqkpCR9/fXXdX5WVFSkhIQEffLJJ8rJydHUqVNVWVnJ37oBuGQSAAAD1Nxi/v7771eHDh0MjqbpdejQQffff78kacmSJQZHAwDBg/oB+J7L5dLHH3+sIUOG6JprrtGwYcP01ltv6cCBA5dshknS+fPn622G3XjjjYqPj5fkeTfK1157rcG7UcJ32OMAABigZm6U4cOHGxyJ79S8tg8//NDgSAAgeFA/AN/aunWrOnfurLS0NOXk5Ki0tLTJtr1371517txZW7duda/72c9+pqFDhzbZc8B7Qd0Q+9e//qXXX39dffr00f/6X/9LzZo10zXXXKOBAwfqiy++MDo8AIBFnTx5UgcPHpQk3XrrrQZH4zs1r83hcOiHH34wNhgfKy4uls1m8/gXHh6udu3aKT093etbrjdkyJAhstlsKi4ubpqALaKx43L69GnFxcUpIiJCBQUF9W4zOztbNpstqJsRMC/qR/ChfpjHv//9bz388MPq3bu3/vnPf17y8QkJCUpNTVXv3r3Vu3dvpaamKiEh4ZK/989//lO9e/fWI488ov/3//5fU4SOyxTUc4gtWLBA2dnZSkhIUJ8+fRQdHa3CwkK99957eu+99/TWW2/poYceMjpMAIDF7NmzR5LUsWNHXX311QZH4ztRUVGKi4tTcXGx9uzZo7vuusvokHwuISFBjz76qCTp7NmzysvLU25urt577z1t3bpVqampBkdoTd6OS6tWrbRq1Sr16tVLTzzxhD777DOFhf30dnn//v2aOXOmOnTooHnz5hnyWmBt1I/gRf0wjsvl0ptvvqlnnnlGp0+frvcxoaGhuu+++/R//s//0Y033qguXbqoVatW9T529OjRWrZs2SWf9z//8z/10Ucf6ZVXXtGoUaOu6DXg8gR1Q6xnz576xz/+oTvuuMNj/c6dO3X33XdrzJgx6t+/f1DdmQUAYH41n/ampKQYHInvde/eXcXFxdq9e7clTmgSExP1/PPPe6zLyspSZmamnnvuOX388cfGBGZxjRmX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"
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "running function 1 of 10\n",
      "Cost at step 1000: 0.04213770610795815\n",
      "Cost at step 2000: 0.039740825771240926\n",
      "Cost at step 3000: 0.040696589860078\n",
      "Cost at step 4000: 0.048459336865629886\n",
      "Cost at step 5000: 0.059556001176232456\n",
      "numParams:  18\n",
      "Approach denseFreqs_parallel_ternary: Run: 0, Cost = 0.0596, R2_Score = 0.5319\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/var/folders/lp/jt0n7jd956q7d8s04306_43r0000gn/T/ipykernel_99421/2270229868.py:111: FutureWarning: The behavior of DataFrame concatenation with empty or all-NA entries is deprecated. In a future version, this will no longer exclude empty or all-NA columns when determining the result dtypes. To retain the old behavior, exclude the relevant entries before the concat operation.\n",
      "  metrics_df = pd.concat([metrics_df, pd.DataFrame({\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Cost at step 1000: 0.05111617695072215\n",
      "Cost at step 2000: 0.03524990477980734\n",
      "Cost at step 3000: 0.05152945705897643\n",
      "Cost at step 4000: 0.04343909609722682\n",
      "Cost at step 5000: 0.050637742464233394\n",
      "numParams:  18\n",
      "Approach denseFreqs_parallel_ternary: Run: 1, Cost = 0.0506, R2_Score = 0.5313\n",
      "Cost at step 1000: 0.05311286881947446\n",
      "Cost at step 2000: 0.051806171949102656\n",
      "Cost at step 3000: 0.03974191679618765\n",
      "Cost at step 4000: 0.05499568451258209\n",
      "Cost at step 5000: 0.04295868227368587\n",
      "numParams:  18\n",
      "Approach denseFreqs_parallel_ternary: Run: 2, Cost = 0.0430, R2_Score = 0.5366\n",
      "Cost at step 1000: 0.08741960220528133\n",
      "Cost at step 2000: 0.06751416711979648\n",
      "Cost at step 3000: 0.046059055008120894\n",
      "Cost at step 4000: 0.05101006547676965\n",
      "Cost at step 5000: 0.05533224846928556\n",
      "numParams:  18\n",
      "Approach denseFreqs_parallel_ternary: Run: 3, Cost = 0.0553, R2_Score = 0.5439\n",
      "Cost at step 1000: 0.11696097002222272\n",
      "Cost at step 2000: 0.06761885913772668\n",
      "Cost at step 3000: 0.04088056193850789\n",
      "Cost at step 4000: 0.060735040155343235\n",
      "Cost at step 5000: 0.03782923823703202\n",
      "numParams:  18\n",
      "Approach denseFreqs_parallel_ternary: Run: 4, Cost = 0.0378, R2_Score = 0.5211\n",
      "Cost at step 1000: 0.06964046289405179\n",
      "Cost at step 2000: 0.04543392138241972\n",
      "Cost at step 3000: 0.053412254820365816\n",
      "Cost at step 4000: 0.061807429606688016\n",
      "Cost at step 5000: 0.050218504300750436\n",
      "numParams:  18\n",
      "Approach denseFreqs_parallel_ternary: Run: 5, Cost = 0.0502, R2_Score = 0.5533\n",
      "Cost at step 1000: 0.1491380499691732\n",
      "Cost at step 2000: 0.06799802235503048\n",
      "Cost at step 3000: 0.07720212378506125\n",
      "Cost at step 4000: 0.07107111370319386\n",
      "Cost at step 5000: 0.050369145767655436\n",
      "numParams:  18\n",
      "Approach denseFreqs_parallel_ternary: Run: 6, Cost = 0.0504, R2_Score = 0.2629\n",
      "Cost at step 1000: 0.10697679370807996\n",
      "Cost at step 2000: 0.05468376158865504\n",
      "Cost at step 3000: 0.03741773694601163\n",
      "Cost at step 4000: 0.057672441785357687\n",
      "Cost at step 5000: 0.0440984583241907\n",
      "numParams:  18\n",
      "Approach denseFreqs_parallel_ternary: Run: 7, Cost = 0.0441, R2_Score = 0.5264\n",
      "Cost at step 1000: 0.040943090725572985\n",
      "Cost at step 2000: 0.043052317252515765\n",
      "Cost at step 3000: 0.044585261038549734\n",
      "Cost at step 4000: 0.04367274374750352\n",
      "Cost at step 5000: 0.055535670977775634\n",
      "numParams:  18\n",
      "Approach denseFreqs_parallel_ternary: Run: 8, Cost = 0.0555, R2_Score = 0.5453\n",
      "Cost at step 1000: 0.050228872958176955\n",
      "Cost at step 2000: 0.06259856506076293\n",
      "Cost at step 3000: 0.04868822985146792\n",
      "Cost at step 4000: 0.05022894419724812\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n",
      "KeyboardInterrupt\n",
      "\n"
     ]
    }
   ],
   "execution_count": 12
  },
  {
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-09-16T10:40:28.105534Z",
     "start_time": "2025-09-16T10:40:28.099372Z"
    }
   },
   "cell_type": "code",
   "source": [],
   "outputs": [],
   "execution_count": null
  }
 ],
 "metadata": {
  "kernelspec": {
   "name": "python3",
   "language": "python",
   "display_name": "Python 3 (ipykernel)"
  },
  "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.15"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 0
}
