{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Import Libraries\n",
    "import numpy as np\n",
    "import paddle\n",
    "import paddle_quantum\n",
    "from paddle_quantum.ansatz import Circuit\n",
    "from paddle_quantum.state import zero_state\n",
    "from numpy import pi as PI\n",
    "from paddle import matmul, transpose, reshape\n",
    "from paddle_quantum.qinfo import pauli_str_to_matrix\n",
    "from paddle_quantum.linalg import dagger\n",
    "from paddle_quantum.dataset import *\n",
    "from paddle_quantum.loss import ExpecVal\n",
    "\n",
    "from matplotlib import pyplot as plt\n",
    "import time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def Observable(n):\n",
    "    r\"\"\"\n",
    "    :param n: bit number\n",
    "    :return: local observable: Z \\otimes I \\otimes ...\\otimes I\n",
    "    \"\"\"\n",
    "    Ob = pauli_str_to_matrix([[1.0, 'x0']], n)\n",
    "\n",
    "    return Ob"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Opt_Classifier(paddle_quantum.Operator):\n",
    "    def __init__(self, n, depth,  seed_paras=1):\n",
    "\n",
    "        super(Opt_Classifier, self).__init__()\n",
    "        self.n = n\n",
    "        self.depth = depth\n",
    "        paddle.seed(seed_paras)\n",
    "        \n",
    "        self.para = self.create_parameter(\n",
    "            shape=[(depth+3)*n],\n",
    "            default_initializer=paddle.nn.initializer.Uniform(0, 2*np.pi),\n",
    "            dtype='float32')        \n",
    "\n",
    "\n",
    "    # foward, predict, loss\n",
    "    def forward(self, data, label):\n",
    "        \"\"\"\n",
    "        input: data: input data unitary, shape: [BATCH, 1, 2^n]\n",
    "               label: shape: [BATCH, 1]\n",
    "        \"\"\"\n",
    "        Ob = paddle.to_tensor(Observable(self.n))\n",
    "        label_pp = reshape(paddle.to_tensor(label), [-1, 1])\n",
    "        All_data = paddle.concat(data, axis=0)\n",
    "        state_in = reshape(zero_state(num_qubits=self.n).data, (-1, 1, 2**self.n))\n",
    "        state_out = state_in\n",
    "        count = 0\n",
    "\n",
    "        # set up circuit\n",
    "        for _ in range(self.depth):\n",
    "            circuit = Circuit(self.n)    \n",
    "            circuit.ry([k for k in range(0, self.n)], param=self.para[count: count + self.n])                       \n",
    "            circuit.cnot()    \n",
    "            count += self.n\n",
    "            state_out = matmul(state_out, matmul(circuit.unitary_matrix().unsqueeze(0), All_data))\n",
    "\n",
    "        E_Z = matmul(matmul(state_out, Ob), transpose(paddle.conj(state_out), perm=[0, 2, 1]))      \n",
    "        state_predict = paddle.real(E_Z)[:, 0] * 0.5 + 0.5\n",
    "\n",
    "        loss = paddle.mean((state_predict - label_pp) ** 2)\n",
    "        is_correct = (paddle.abs(state_predict - label_pp) < 0.5).nonzero().shape[0]\n",
    "        acc = is_correct / label.shape[0]\n",
    "\n",
    "        return loss, acc, state_predict.numpy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def QClassifier(quantum_train_x, train_y, quantum_test_x, test_y, N, DEPTH, EPOCH, L, seed):\n",
    "\n",
    "    net = Opt_Classifier(n=N, depth=DEPTH, seed_paras=seed)\n",
    "    summary_iter, summary_test_acc, summary_train_acc = [], [], []\n",
    " \n",
    "    # SGD \n",
    "    opt = paddle.optimizer.SGD(learning_rate=L, parameters=net.parameters())\n",
    "    \n",
    "\n",
    "    # optimize\n",
    "    for ep in range(EPOCH):        \n",
    "\n",
    "        loss, train_acc, state_predict_useless= net(data=quantum_train_x ,label=train_y)\n",
    "        loss_useless, test_acc, state_predict_useless = net(data=quantum_test_x ,label=test_y)\n",
    "        loss.backward()\n",
    "        opt.minimize(loss)\n",
    "        opt.clear_grad()\n",
    "\n",
    "        if ep % 50 == 0:\n",
    "            print(\"epoch:\", ep,\n",
    "                    \"loss: %.4f\" % loss.numpy(),\n",
    "                    \"train acc: %.4f\" % train_acc,\n",
    "                    \"test acc: %.4f\" % test_acc)\n",
    "        \n",
    "        summary_train_acc.append(loss.numpy())  \n",
    "        summary_test_acc.append(loss_useless.numpy())\n",
    "        \n",
    "    return summary_test_acc, summary_train_acc"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "v = 5\n",
    "ita = 0.01\n",
    "ep = 50\n",
    "seed = np.random.randint(0, high=1e5, size=[v], dtype=int)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "data = BreastCancer(encoding='angle_encoding', num_qubits=4, test_rate=0.8, return_state=True, seed=0)\n",
    "    \n",
    "quantum_train_x, train_y = data.train_circuits, data.train_y\n",
    "quantum_test_x, test_y = data.test_circuits, data.test_y"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "train_x = []\n",
    "for i in range(0, len(train_y)):\n",
    "    cir_train = Circuit(4)\n",
    "    cir_train.extend(quantum_train_x[i][0])\n",
    "    train_x.append(cir_train.unitary_matrix().unsqueeze(0))\n",
    "\n",
    "test_x = []\n",
    "for i in range(len(test_y)):\n",
    "    cir_test = Circuit(4)\n",
    "    cir_test.extend(quantum_test_x[i][0])\n",
    "    test_x.append(cir_test.unitary_matrix().unsqueeze(0))  "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "train_data_L2, test_data_L2 = [], []\n",
    "train_data_L8, test_data_L8 = [], []\n",
    "train_data_L16, test_data_L16 = [], []\n",
    "\n",
    "for trial in range(v):\n",
    "    \n",
    "    te_acc, tr_acc = QClassifier(\n",
    "      train_x,\n",
    "      train_y,\n",
    "      test_x,\n",
    "      test_y,\n",
    "      N = 4,\n",
    "      DEPTH = 2,\n",
    "      EPOCH = ep,\n",
    "      L = ita,\n",
    "      seed=seed[trial]\n",
    "    )\n",
    "\n",
    "    train_data_L2.append(tr_acc)\n",
    "    test_data_L2.append(te_acc)\n",
    "  \n",
    "\n",
    "    te_acc, tr_acc = QClassifier(\n",
    "      train_x,\n",
    "      train_y,\n",
    "      test_x,\n",
    "      test_y,\n",
    "      N = 4,\n",
    "      DEPTH = 8,\n",
    "      EPOCH = ep,\n",
    "      L = ita,\n",
    "      seed=seed[trial]\n",
    "    )\n",
    "\n",
    "    train_data_L8.append(tr_acc)\n",
    "    test_data_L8.append(te_acc)\n",
    "    \n",
    "    te_acc, tr_acc = QClassifier(\n",
    "      train_x,\n",
    "      train_y,\n",
    "      test_x,\n",
    "      test_y,\n",
    "      N = 4,\n",
    "      DEPTH = 16,\n",
    "      EPOCH = ep,\n",
    "      L = ita,\n",
    "      seed=seed[trial]\n",
    "    )\n",
    "\n",
    "    train_data_L16.append(tr_acc)\n",
    "    test_data_L16.append(te_acc)\n",
    "    "
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "paddle_quantum_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.8.18"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
