{
 "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, 'z0']], 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",
    "    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",
    "        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, LR, seed):\n",
    "    net = Opt_Classifier(n=N, depth=DEPTH, seed_paras=seed)\n",
    "\n",
    "    summary_iter, summary_test_acc, summary_train_acc = [], [], []\n",
    "    summary_test_loss, summary_train_loss = [], []\n",
    "\n",
    "    # SGD \n",
    "    opt = paddle.optimizer.SGD(learning_rate=LR, parameters=net.parameters())\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_loss.append(loss[0].item())  \n",
    "        summary_test_loss.append(loss_useless[0].item())\n",
    "        \n",
    "        summary_train_acc.append(train_acc)  \n",
    "        summary_test_acc.append(test_acc)       \n",
    "\n",
    "    return summary_train_loss, summary_test_loss, summary_train_acc, summary_test_acc"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "v = 5\n",
    "ita = 0.01\n",
    "ep = 1000\n",
    "seed = np.random.randint(0, high=1e5, size=[v], dtype=int)\n",
    "print(seed)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def test_acc(testing_data_num, classes):\n",
    "    num_qubit = 4\n",
    "\n",
    "    val_dataset = MNIST(mode='test', encoding='angle_encoding', num_qubits=num_qubit, classes=classes,\n",
    "                        data_num=testing_data_num,need_cropping=True,\n",
    "                        downscaling_method='resize', target_dimension=16,return_state=True, seed=0)\n",
    "\n",
    "    quantum_test_x, test_circuit, test_y = val_dataset.quantum_image_states, val_dataset.quantum_image_circuits,val_dataset.labels\n",
    "    \n",
    "    return quantum_test_x, test_circuit, test_y\n",
    "\n",
    "\n",
    "def train_acc(training_data_num, classes):\n",
    "    num_qubit = 4\n",
    "\n",
    "    train_accset = MNIST(mode='train', encoding='angle_encoding', num_qubits=num_qubit, classes=classes,\n",
    "                        data_num=training_data_num,need_cropping=True,\n",
    "                        downscaling_method='resize', target_dimension=16, return_state=True, seed=0)\n",
    "\n",
    "    quantum_train_x, train_circuit, train_y = train_accset.quantum_image_states, train_accset.quantum_image_circuits, train_accset.labels\n",
    "\n",
    "    \n",
    "    return quantum_train_x, train_circuit, train_y"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "quantum_test_x, test_circuit, test_y = test_acc(2000, [0, 1]) \n",
    "quantum_train_x, train_circuit, train_y = train_acc(500, [0, 1]) "
   ]
  },
  {
   "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(train_circuit[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(test_circuit[i][0])\n",
    "    test_x.append(cir_test.unitary_matrix().unsqueeze(0))  "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "train_acc_L2, test_acc_L2 = [], []\n",
    "train_acc_L8, test_acc_L8 = [], []\n",
    "train_acc_L16, test_acc_L16 = [], []\n",
    "\n",
    "train_loss_L2, test_loss_L2 = [], []\n",
    "train_loss_L8, test_loss_L8 = [], []\n",
    "train_loss_L16, test_loss_L16 = [], []\n",
    "\n",
    "for trial in range(v):\n",
    "\n",
    "    tr_loss, te_loss, tr_acc, te_acc = QClassifier(\n",
    "      train_x,\n",
    "      train_y,\n",
    "      test_x,\n",
    "      test_y,\n",
    "      N = 4,\n",
    "      DEPTH = 2,\n",
    "      EPOCH = ep,\n",
    "      LR = ita,\n",
    "      seed=seed[trial]\n",
    "    )\n",
    "\n",
    "    train_acc_L2.append(tr_acc)\n",
    "    test_acc_L2.append(te_acc)\n",
    "    \n",
    "    train_loss_L2.append(tr_loss)\n",
    "    test_loss_L2.append(te_loss)\n",
    "  \n",
    "\n",
    "    tr_loss, te_loss, tr_acc, te_acc = QClassifier(\n",
    "      train_x,\n",
    "      train_y,\n",
    "      test_x,\n",
    "      test_y,\n",
    "      N = 4,\n",
    "      DEPTH = 8,\n",
    "      EPOCH = ep,\n",
    "      LR = ita,\n",
    "      seed=seed[trial]\n",
    "    )\n",
    "\n",
    "    train_acc_L8.append(tr_acc)\n",
    "    test_acc_L8.append(te_acc)\n",
    "    \n",
    "    train_loss_L8.append(tr_loss)\n",
    "    test_loss_L8.append(te_loss)\n",
    "    \n",
    "    tr_loss, te_loss, tr_acc, te_acc = QClassifier(\n",
    "      train_x,\n",
    "      train_y,\n",
    "      test_x,\n",
    "      test_y,\n",
    "      N = 4,\n",
    "      DEPTH = 16,\n",
    "      EPOCH = ep,\n",
    "      LR = ita,\n",
    "      seed=seed[trial]\n",
    "    )\n",
    "\n",
    "    train_acc_L16.append(tr_acc)\n",
    "    test_acc_L16.append(te_acc)\n",
    "    \n",
    "    train_loss_L16.append(tr_loss)\n",
    "    test_loss_L16.append(te_loss)\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
}
