{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "e8a46331",
   "metadata": {},
   "source": [
    "# Simulations for EAGLE\n",
    "This notebook provides code to reproduce the EAGLE performance plots.\n",
    "Parameters can be modified in the last cell."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "5c0d8470",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import time\n",
    "import math\n",
    "import matplotlib.pyplot as plt\n",
    "import pandas as pd\n",
    "import seaborn as sns\n",
    "\n",
    "eps32 = np.finfo(np.float32).eps    # tiny floor for numerics\n",
    "\n",
    "plt.rcParams.update({\n",
    "    'axes.linewidth': 4,  # Thicker axes\n",
    "    'xtick.major.width': 4,  # Thicker x-axis ticks\n",
    "    'ytick.major.width': 4,  # Thicker y-axis ticks\n",
    "    'lines.linewidth': 4,  # Thicker plot lines\n",
    "    'font.size': 18  # Larger font size\n",
    "})"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a8d148b1",
   "metadata": {},
   "source": [
    "### Solvers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "24bad8cf",
   "metadata": {},
   "outputs": [],
   "source": [
    "def solve_gdpp(A_, B_, C, s, skip_m, prec=1e-20, max_iter=500):\n",
    "    \n",
    "    start = time.perf_counter()\n",
    "    \n",
    "    def spectral_norm(X):\n",
    "        v = np.random.randn(X.shape[1],1)\n",
    "        for _ in range(2):\n",
    "            v  = X.T  @ (X @ v)\n",
    "            v /= np.linalg.norm(v) + eps32\n",
    "        return float(np.linalg.norm(X @ v))  \n",
    "    \n",
    "    # ------ pre-normalise each worker block ---------------------------------------\n",
    "    alpha = [spectral_norm(A_i)                               for A_i in A_]\n",
    "    A_    = [A_i / a                           for A_i, a in zip(A_, alpha)]\n",
    "    B_    = [B_i / a                           for B_i, a in zip(B_, alpha)]\n",
    "\n",
    "    # -------------- allocate memory -------------------------------------------------\n",
    "    D = np.zeros((B_[0].shape[0], C.shape[1]))\n",
    "    dA = np.zeros_like(A_[0]) \n",
    "    dB = np.zeros_like(B_[0])\n",
    "    C = C.copy()\n",
    "    D = D.copy()\n",
    "    C_acc = np.zeros_like(C)\n",
    "    D_acc = np.zeros_like(D)\n",
    "\n",
    "    # -------------- main loop -----------------------------------------------------\n",
    "    m = len(A_)\n",
    "    iters = np.ones((max_iter, *D.shape))  \n",
    "    times = np.ones(max_iter)      \n",
    "    for k in range(max_iter):\n",
    "        iters[k] = D.copy()\n",
    "        times[k] = time.perf_counter() - start\n",
    "        if (k > 0 and np.linalg.norm(iters[k-1]-iters[k]) < prec) or k>=max_iter:\n",
    "            break\n",
    "\n",
    "        \n",
    "        C_acc.fill(0)\n",
    "        D_acc.fill(0)\n",
    "\n",
    "        for i in range(m):\n",
    "            Ai, Bi = A_[i], B_[i]\n",
    "            \n",
    "            if s >= A_[0].shape[1]: # no sketch\n",
    "                \n",
    "                lam  = spectral_norm(Ai)       \n",
    "                rhot = 1.0 / (3.0 * max(lam*lam, eps32))\n",
    "\n",
    "                ASAST = Ai @ Ai.T              \n",
    "                BSAST = Bi @ Ai.T\n",
    "\n",
    "                B_[i] -= rhot * BSAST @ Ai\n",
    "                A_[i] -= rhot * ASAST @ Ai\n",
    "                dC = 3*rhot * ASAST @ C\n",
    "                dD = 3*rhot * BSAST @ C\n",
    "                \n",
    "            else: # sketch\n",
    "                \n",
    "                idx = np.random.choice(Ai.shape[1], size=s, replace=False)\n",
    "                AS = Ai[:,idx]\n",
    "                BS = Bi[:,idx]\n",
    "                \n",
    "                lam  = spectral_norm(AS)       \n",
    "                rhot = 1.0 / (3.0 * max(lam*lam, eps32))\n",
    "\n",
    "                ASTAS =  AS.T @ AS             \n",
    "\n",
    "                dA.fill(0); dB.fill(0)\n",
    "                \n",
    "                dD = 3*rhot * (BS @ AS.T) @ C\n",
    "                dC = 3*rhot * AS @ (AS.T @ C)\n",
    "                B_[i][:,idx] -= rhot * BS @ ASTAS\n",
    "                A_[i][:,idx] -= rhot * AS @ ASTAS\n",
    "\n",
    "            g      = max(spectral_norm(A_[i]), eps32)\n",
    "            A_[i] /= g\n",
    "            B_[i] /= g\n",
    "            alpha[i] *= g\n",
    "\n",
    "            C_acc += dC\n",
    "            D_acc += dD\n",
    "\n",
    "        if skip_m:\n",
    "            C     -= C_acc          \n",
    "            D     += D_acc \n",
    "        else:\n",
    "            C     -= C_acc / m           \n",
    "            D     += D_acc / m\n",
    "\n",
    "    iters[k+1:] = iters[k]\n",
    "    return iters, times[:k+1]\n",
    "\n",
    "def solve_lstsq(A_, B_, C):\n",
    "    start = time.perf_counter()\n",
    "    assert len(A_) == 1\n",
    "    A, B = A_[0], B_[0]\n",
    "    X, *_ = np.linalg.lstsq(A, C, rcond=None)\n",
    "    D = B @ X\n",
    "    times = time.perf_counter() - start\n",
    "    return [np.zeros_like(D), D], np.array([0, times])\n",
    "\n",
    "\n",
    "def solve_gd(\n",
    "        A_, B_,           \n",
    "        C,                \n",
    "        s,                \n",
    "        rng=None,\n",
    "        prec=1e-20,\n",
    "        max_iter=10000):\n",
    "    if rng is None:\n",
    "        rng = np.random.default_rng()\n",
    "    eps = 1e-12\n",
    "\n",
    "    # shapes\n",
    "    p_A, n_tot  = A_[0].shape        # rows of A, cols per worker block\n",
    "    p_B         = B_[0].shape[0]     # rows of B  (=> rows of X)\n",
    "    m_out       = C.shape[1]\n",
    "    m_workers   = len(A_)\n",
    "\n",
    "    # variables\n",
    "    X       = np.zeros((p_B, p_A))        # parameter we learn\n",
    "    iters   = np.empty((max_iter, p_B, m_out)) \n",
    "    elapsed = np.empty(max_iter)\n",
    "\n",
    "    # helper: σ_max² by two power iterations  (no big allocs)\n",
    "    def sigma_max_sq(mat):\n",
    "        v = rng.standard_normal((mat.shape[1], 1))\n",
    "        v /= np.linalg.norm(v) + eps\n",
    "        for _ in range(2):\n",
    "            v  = mat.T @ (mat @ v)\n",
    "            v /= np.linalg.norm(v) + eps\n",
    "        return float(np.linalg.norm(mat @ v) ** 2)\n",
    "\n",
    "    t0 = time.perf_counter()\n",
    "\n",
    "    for k in range(max_iter):\n",
    "        # --- log completed block & clock ------------------------------\n",
    "        iters[k]   = X @ C\n",
    "        elapsed[k] = time.perf_counter() - t0\n",
    "        if (k>0 and np.linalg.norm(iters[k-1]-iters[k]) < prec) or k>=max_iter:\n",
    "            break\n",
    "\n",
    "        # --- gradient accumulation -----------------------------------\n",
    "        G      = np.zeros_like(X)\n",
    "        if s < A_[0].shape[1] or k == 0:\n",
    "            max_L  = 0.0                   # Lipschitz across workers\n",
    "\n",
    "        for Ai, Bi in zip(A_, B_):\n",
    "\n",
    "            # -- optional column sketch --------------------------------\n",
    "            n_i = Ai.shape[1]\n",
    "            if s < n_i:\n",
    "                idx  = rng.choice(n_i, size=s, replace=False)\n",
    "                A_s  = Ai[:, idx]          \n",
    "                B_s  = Bi[:, idx]          \n",
    "            else:\n",
    "                A_s, B_s = Ai, Bi\n",
    "\n",
    "            # gradient for this worker\n",
    "            EA   = X @ A_s - B_s           \n",
    "            G   += EA @ A_s.T               \n",
    "            \n",
    "            if s < n_i or k == 0:\n",
    "                # track Lipschitz constant \n",
    "                max_L = max(max_L, sigma_max_sq(A_s))\n",
    "\n",
    "        # --- global safe step ----------------------------------------\n",
    "        eta = 1.0 / (max_L + eps)          \n",
    "\n",
    "        # --- single synchronous update --------------------------------\n",
    "        X -= (eta / m_workers) * G\n",
    "        \n",
    "\n",
    "    iters[k+1:] = iters[k]\n",
    "    return iters, elapsed[:k+1]\n",
    "\n",
    "\n",
    "def solve_cg(\n",
    "        A_, B_, C,\n",
    "        s=None,              \n",
    "        rng=None,\n",
    "        prec=1e-20,\n",
    "        max_iter=10000):\n",
    "    \"\"\"\n",
    "    CG with optional column sketching.\n",
    "    \"\"\"\n",
    "    if rng is None:\n",
    "        rng = np.random.default_rng()\n",
    "\n",
    "    p_A, n = A_[0].shape\n",
    "    p_B, _ = B_[0].shape\n",
    "    m_C    = C.shape[1]\n",
    "    m_wrk  = len(A_)\n",
    "\n",
    "    # --- sketching -----------------------------------------------------\n",
    "    def sketch(A, B):\n",
    "        if s is not None and s < n:\n",
    "            idx = rng.choice(n, size=s, replace=False)\n",
    "            return A[:, idx], B[:, idx]\n",
    "        return A, B\n",
    "\n",
    "    G = np.zeros((p_A, p_A))\n",
    "    B_rhs = np.zeros((p_B, p_A))\n",
    "    for Ai, Bi in zip(A_, B_):\n",
    "        Ai_s, Bi_s = sketch(Ai, Bi)\n",
    "        G += Ai_s @ Ai_s.T\n",
    "        B_rhs += Bi_s @ Ai_s.T\n",
    "    G /= m_wrk\n",
    "    B_rhs /= m_wrk\n",
    "\n",
    "    # --- CG initialisation --------------------------------------------\n",
    "    X   = np.zeros_like(B_rhs)\n",
    "    R   = B_rhs.copy()\n",
    "    P   = R.copy()\n",
    "    rr  = np.sum(R*R)\n",
    "\n",
    "    iters   = np.empty((max_iter, p_B, m_C))\n",
    "    elapsed = np.empty(max_iter)\n",
    "\n",
    "    start = time.perf_counter()\n",
    "\n",
    "    for k in range(max_iter):\n",
    "        iters[k]   = X @ C\n",
    "        elapsed[k] = time.perf_counter() - start\n",
    "        if (k>0 and np.linalg.norm(iters[k-1]-iters[k]) < prec) or k>=max_iter:\n",
    "            break\n",
    "\n",
    "        GP   = P @ G\n",
    "        alpha = rr / np.sum(P * GP)\n",
    "        X    += alpha * P\n",
    "        R    -= alpha * GP\n",
    "        rr_new = np.sum(R*R)\n",
    "\n",
    "        beta  = rr_new / rr\n",
    "        P     = R + beta * P\n",
    "        rr    = rr_new\n",
    "\n",
    "    iters[k+1:] = iters[k]\n",
    "    return iters, elapsed[:k+1]"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6ca3c2ba",
   "metadata": {},
   "source": [
    "### Data generation (controls data diversity in distributed setting)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "b58b8fb4",
   "metadata": {},
   "outputs": [],
   "source": [
    "def _geometric_spectrum(r: int, cond: float) -> np.ndarray:\n",
    "    \"\"\"\n",
    "    r singular values with ratio σ₁/σᵣ = cond (largest = 1, smallest = 1/cond).\n",
    "    Geometric spacing avoids a big ‘gap’ in the middle.\n",
    "    \"\"\"\n",
    "    return np.power(10., (-np.linspace(0.0, cond, r)))\n",
    "\n",
    "def _orth_rows(n_rows: int, n_cols: int, rng: np.random.Generator) -> np.ndarray:\n",
    "    \"\"\"\n",
    "    Random matrix with *orthonormal rows* (shape n_rows × n_cols, n_cols ≥ n_rows).\n",
    "    \"\"\"\n",
    "    Q, _ = np.linalg.qr(rng.standard_normal((n_cols, n_rows)))  # orthonormal columns\n",
    "    return Q.T                                                  # orthonormal rows\n",
    "\n",
    "def _wrap_slice(mat: np.ndarray, start: int, width: int) -> np.ndarray:\n",
    "    \"\"\"Take `width` columns of `mat`, wrapping around if necessary.\"\"\"\n",
    "    idx = (np.arange(width) + start) % mat.shape[1]\n",
    "    return mat[:, idx]\n",
    "\n",
    "# ----------------------------------------------------------------------------- #\n",
    "def make_Z(d: int, n: int, r: int, m: int, alpha: float, cond: float,\n",
    "           rng: np.random.Generator = None) -> np.ndarray:\n",
    "    \"\"\"\n",
    "    Return Z ∈ ℝ^{d×n} split into m vertical blocks.\n",
    "    * alpha = 0 → identical column spaces\n",
    "    * alpha = 1 → maximally independent column spaces\n",
    "    * Each block has rank r and condition number `cond`.\n",
    "    \"\"\"\n",
    "    if rng is None:\n",
    "        rng = np.random.default_rng()\n",
    "\n",
    "    assert 0.0 <= alpha <= 1.0\n",
    "    assert cond >= 1.0, \"condition number must be ≥ 1\"\n",
    "    assert r <= d,      \"rank r cannot exceed d\"\n",
    "\n",
    "    # (1) Shared r-dimensional basis, common to every block (for the overlap part)\n",
    "    U_shared, _ = np.linalg.qr(rng.standard_normal((d, r)))\n",
    "\n",
    "    # (2) Large orthonormal pool to harvest extra directions\n",
    "    V_pool, _   = np.linalg.qr(rng.standard_normal((d, d)))\n",
    "\n",
    "    # (3) Block widths — allow n not divisible by m\n",
    "    base = n // m\n",
    "    block_ns = [base] * m\n",
    "    for i in range(n % m):\n",
    "        block_ns[i] += 1\n",
    "\n",
    "    # (4) Fixed singular spectrum (same for every block) with the wanted cond #\n",
    "    sing = _geometric_spectrum(r, cond)              # length r, σ₁/σᵣ = cond\n",
    "    Σ     = np.diag(sing)\n",
    "\n",
    "    blocks = []\n",
    "    for j, n_j in enumerate(block_ns):\n",
    "        if n_j < r:\n",
    "            raise ValueError(\n",
    "                f\"Block {j} has only {n_j} columns < r={r}; \"\n",
    "                \"cannot reach full rank.\"\n",
    "            )\n",
    "\n",
    "        # ---- Block-specific left basis:  blend, then re-orthonormalise ---- #\n",
    "        V_unique = _wrap_slice(V_pool, start=j * r, width=r)     # r columns\n",
    "        V_mix    = alpha * V_unique + (1 - alpha) * U_shared\n",
    "        B_j, _   = np.linalg.qr(V_mix)                           # d × r\n",
    "\n",
    "        # ---- Right part with orthonormal rows so σ = sing exactly ---- #\n",
    "        Q_j = _orth_rows(r, n_j, rng)                            # r × n_j, Q_j Q_jᵀ = I_r\n",
    "\n",
    "        # final block:  B_j · Σ · Q_j    → rank r, cond = cond\n",
    "        blocks.append(B_j @ Σ @ Q_j)\n",
    "\n",
    "    Z = np.concatenate(blocks, axis=1)                           # d × n\n",
    "    return Z"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a1049086",
   "metadata": {},
   "source": [
    "### Benchmark"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "b376c3f5",
   "metadata": {},
   "outputs": [],
   "source": [
    "def run(bsize=50, n=1001, d=1001, p=2, r=99, s=99, m=1, eps=0., alpha=1., cond=2, prec=1e-20, dat=\"orthogonal\"):\n",
    "\n",
    "    solvers = {\n",
    "        \"gdpp\": lambda A_, B_, C: solve_gdpp(A_, B_, C, s=s, skip_m=((m>1) and (alpha==1)), max_iter=300, prec=prec),\n",
    "        \"gd\": lambda A_, B_, C: solve_gd( A_, B_, C, s=s, max_iter=10000, prec=prec),\n",
    "    }\n",
    "    if m == 1 and s==n-p:\n",
    "        solvers[\"lstsq\"] = solve_lstsq\n",
    "    if m==1 and r == (n-p) and s==n-p:\n",
    "        solvers[\"cg\"] = lambda A_, B_, C: solve_cg( A_, B_, C, s=s, max_iter=10000, prec=prec)\n",
    "\n",
    "    A_ = [None]*m\n",
    "    B_ = [None]*m\n",
    "\n",
    "    data = []\n",
    "    for k in range(bsize):\n",
    "        \n",
    "        if dat == \"orthogonal\":\n",
    "            a = make_Z(d-p, n-p, r//m, m, alpha, cond)\n",
    "            a = np.concatenate([a, a@np.random.randn(n-p, p)], axis=1)\n",
    "            b = np.random.randn(p,d-p)\n",
    "            z = np.concatenate([a, b@a], axis=0)\n",
    "        elif dat == \"low-rank\":\n",
    "            z = np.random.randn(d,r) @ np.random.randn(n,r).T / math.sqrt(r)\n",
    "        elif dat == \"student\":\n",
    "            ν = 4                       # degrees of freedom (heavier tails if smaller)\n",
    "            A = np.random.standard_t(ν, size=(d, r))\n",
    "            B = np.random.standard_t(ν, size=(n, r))\n",
    "            # Match variance of N(0,1) --> divide by sqrt(ν / (ν-2))  (valid for ν>2)\n",
    "            scale = math.sqrt(ν / (ν - 2))\n",
    "            z = (A / scale) @ (B / scale).T / math.sqrt(r)\n",
    "        elif dat == \"cor_gaus\":\n",
    "            ρ = 0.8\n",
    "            # Build Toeplitz covariance for rows (size d) and columns (size n)\n",
    "            Σ_d = ρ ** np.abs(np.subtract.outer(np.arange(d), np.arange(d)))\n",
    "            Σ_n = ρ ** np.abs(np.subtract.outer(np.arange(n), np.arange(n)))\n",
    "\n",
    "            # Draw r factor columns that each share the same row-covariance\n",
    "            A = np.random.multivariate_normal(np.zeros(d), Σ_d, size=r).T          # d × r\n",
    "            B = np.random.multivariate_normal(np.zeros(n), Σ_n, size=r).T          # n × r\n",
    "\n",
    "            z = A @ B.T / math.sqrt(r)\n",
    "        elif dat == \"rademacher\":\n",
    "            prob = 0.1                          # fraction of non-zeros\n",
    "            # Helper: draw {-1,+1} with equal prob\n",
    "            signsA = np.random.choice((-1.0, 1.0), size=(d, r))\n",
    "            signsB = np.random.choice((-1.0, 1.0), size=(n, r))\n",
    "\n",
    "            maskA  = np.random.rand(d, r) < prob\n",
    "            maskB  = np.random.rand(n, r) < prob\n",
    "\n",
    "            A = (signsA * maskA) / math.sqrt(prob)        # d × r\n",
    "            B = (signsB * maskB) / math.sqrt(prob)        # n × r\n",
    "\n",
    "            z = A @ B.T / math.sqrt(r)\n",
    "        elif dat == \"block\":\n",
    "            k = 5                        # number of row-clusters\n",
    "            # --- Build factor A: one-hot cluster assignments (d × k)\n",
    "            cluster_ids = np.random.randint(0, k, size=d)\n",
    "            A = np.zeros((d, k))\n",
    "            A[np.arange(d), cluster_ids] = 1.0          # each row picks one cluster\n",
    "\n",
    "            # --- Build factor B: cluster centroids (n × k)\n",
    "            B = np.random.randn(n, k)                   # each column is a centroid\n",
    "\n",
    "            z = A @ B.T / math.sqrt(k)                  # note: k used, not r\n",
    "        \n",
    "        \n",
    "        if eps > 0:\n",
    "            z += eps*np.random.randn(*z.shape)\n",
    "\n",
    "        A = z[:-p,:-p]\n",
    "        B = z[-p:,:-p]\n",
    "        C = z[:-p,-p:]\n",
    "        target = z[-p:,-p:]\n",
    "            \n",
    "        # distribute\n",
    "        n_ = (n-p)//m\n",
    "        assert m*n_ == n-p\n",
    "        max_sv = 0\n",
    "        for i in range(m):\n",
    "            A_[i] = np.ascontiguousarray(A[:,i*n_:(i+1)*n_])\n",
    "            B_[i] = np.ascontiguousarray(B[:,i*n_:(i+1)*n_]) \n",
    "            sv = np.linalg.svd(A_[i])[1]\n",
    "            max_sv = max(max_sv, sv[0]/sv[r//m-1])\n",
    "        sv = np.linalg.svd(A)[1]\n",
    "        sv_glob = sv[0]/sv[r-1]   \n",
    "        print(\"Global conditioning of A:\", sv_glob)\n",
    "        print(\"Per-machine conditioning of A:\", max_sv) \n",
    "\n",
    "        for name, solve in solvers.items():\n",
    "            iters, times = solve(A_, B_, C)\n",
    "            for i, (t, loss) in enumerate(zip(times, np.mean(np.square(iters - target[None]), axis=(1,2)))):\n",
    "                data.append({\"solver\": name, \"iteration\": i, \"time\": t, \"error\": loss, \"diversity coefficient\": sv_glob/max_sv})\n",
    "            \n",
    "\n",
    "    data = pd.DataFrame(data)  \n",
    "    \n",
    "    return data"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d6a5e12e",
   "metadata": {},
   "source": [
    "### Plots"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "98dac059",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Global conditioning of A: 100.00000000000034\n",
      "Per-machine conditioning of A: 100.00000000000034\n"
     ]
    },
    {
     "data": {
      "image/png": "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ZdURNhERGRmjm51aFVlZWcqvJRKJxnal60r30cWpU6egUCg0Hs++ZxcbG6uxrTbr4JHh1atXT7mrwnfffYf3339fOSEnKysLK1euRHBwsNoJMB4eHnjzzTcBAJ988glWrFih/LOYmpqK2bNnY8eOHVzsl4iI6kYSR1TT3n//fYwbNw6AsHtD/fr14eTkBEdHRyxYsABTpkzB8OHDAQA2NjYqfVeuXIkBAwZALpfjww8/hKOjI5ycnNCgQQNs3LgRH3zwATp37lzTH4mIiGqZOpHESaVS5dfqHmGpqyvdh6iyLCwssGfPHmzZsgU9e/aEWCxGUVERunfvji1btmDbtm3IyMgAgDLbb9nY2ODQoUNYt24dOnfuDCsrKygUCvj7+2PPnj347LPPav4DERFRrVMn3okrvV5XUlKSxkV2S8/4q2iNr9rGx8cHgDCxITs728DRECDMeJ45cyZmzpxZpk6hUCgXBW7Xrl2ZegsLC7z11lt466231I596tQpvcZKRETGp07ciWvTpg3MzISPWt7szZI6Nzc3vUxqqEklEyfKm31Ltcf27duRmJgICwsLDBgwwNDhEBGREaoTSZxEIoGfnx8A4PDhw2rbKBQKHDlyBIAwu5BIV5MmTcK+ffuQlpamLHv48CE+++wzzJo1CwAQGBgId3d3Q4VIRERGrE48TgWAoKAgnD17FidPnsTFixfRq1cvlfq9e/fi/v37AIR/WIl0dejQIezeLawML5FIYGlpiczMTGW9v78/1q5da6jwiIjIyBldEvfkyRMUFxcrvy9ZxFcmk6nc8bCxsVFZKDUoKAjr1q3DtWvXMHbsWISGhiIgIAByuRxhYWHKOyNDhgwps+UWUVWsX78ehw4dQlRUFB49eoTs7GzUr18fnTt3xsSJEzFt2jRYWloaOkwik8XttcjUGV0S16VLF5XFcEusWrUKq1atUn4fFBSErVu3Kr+3sLBAeHg4+vfvj9jYWAwYMAASiQRyuRx5eXnKsXfu3Fntn4HqhsDAQN7VJSKialMn3okr4e3tjejoaHz88cdo3749RCIRLC0t0a1bN6xevRoXLlxQuwArERERUW1jdHfiYmNjdeovlUqxdOlSLF26VD8BERERERlAnboTR0RERGQqmMQRERERGSEmcURERERGiEkcERERkRFiEkdERERkhJjEERERERkhJnFERERERsjo1okjIiLSRsLrbyDvn39UymzatoXnpq8NFBGRfjGJIyIik1Scno6ihw9Vy9zcDBQNkf7xcSoRERGREWISR2SE+vXrB5FIhODgYEOHQkREBsIkjoiIiMgIMYkjIiIiMkJM4oiIiIiMEJM4IiIiIiPEJI6omigUCoSEhMDX1xdSqRT29vbo1asXNm/eDIVCgenTp0MkEmH69Oll+hYXF+Orr75C165dUa9ePTg5OaFfv37Yt29fhecViUQQiUQ4deoUUlJSMHv2bDRp0gQ2NjZwc3PDlClTcPPmTbV9Y2Njlf1jY2Nx584dTJ8+HY0aNYK1tTUaN26M1157DcnJybpeHiIi0hHXiSOqBsXFxZgyZQp+/PFHAEJi5eDggMjISFy6dAmnTp2ClZWV2r75+fkYOXIkjhw5AgAwMzODlZUVzpw5g9OnT+ODDz7QKoaYmBhMmjQJKSkpEIvFsLS0xMOHD/HDDz/gp59+ws8//4zBgwdr7H/x4kXMmjULWVlZsLW1hbm5ORISEvDtt99i7969OHbsGLp27VrJK0NERPrCJI60tiAsGrceZhk6jGrRqoEUn43tqLfxVq1apUzg5s2bhw8//BDOzs54+vQpNm7ciEWLFsHBwUFt34ULF+LIkSMQiUT45JNPMGfOHNjZ2eHRo0cIDg7GypUrYW9vX2EM77zzDuzt7XH06FEMGDAAIpEIly5dwiuvvIJr165hwoQJ+Pvvv9GoUSO1/V999VU0adIE3333HXr27AmFQoFjx45h1qxZiI+Px+jRo3H9+nVIpdIqXyciIqo6JnGktVsPsxAVn2HoMGq9nJwcrFixAgAwc+ZMrFmzRllnZ2eHhQsXIj8/H0uXLi3TNzk5GV999RUA4KOPPsKiRYuUda6urvj666+RkZGBXbt2VRhHbm4uIiIi0KZNG2VZz549cfz4cbRp0wbp6elYsWIFNm7cqLa/hYUFjh07BldXVwDC3cRBgwbh8OHD6Ny5M+Lj4/HNN99g/vz5WlwVoprntmwZ5LIclTIzST0DRUOkf3wnjkjPjh49iqdPnwKAShJW2rvvvguJRFKmfN++fSgqKoJYLMZ7772ntq+2C/yOHz9eJYEr4erqitdeew0AlHcL1XnttdeUCVxpbdq0wbhx4wAAu3fv1ioWIkOwadUSki5dVA6bVi0NHRaR3jCJI9KzK1euAAAaN26MJk2aqG0jlUrRrVu3MuWRkZEAgO7du8POzk5t35YtW8LDw6PCOJ5//vkK6x4/foyYmJgq94+OjkZhYWGFsRARkf4xiSPSs9TUVABAw4YNy22nLhF79OiRxrrSNL3HVtH46upKzlmV/kVFRUhPT68wFiIi0j++E0daa9XAdF9gr47PJhKJ9D4mERFRCSZxpDV9zt40ZfXr1weACtdSS0pKKlNW8g6aurqK+lamTek6de+9lbRp1apVuf0tLCzg5ORUYSxERKR/fJxKpGcla6fFxcUhNjZWbZvs7Gxcvny5THn37t0BCO/GZWdnq+17584dJCYmVhjHyZMnK6xzcnLS+N6eNv07duwIS0vLCmMhIiL9YxJHpGeDBg1STkr49NNP1bZZu3YtZDJZmfKxY8fC3Nwcubm5WL16tdq+y5Yt0yqOvXv34tatW2XK09LS8O233wIAJkyYoLH/N998g7S0tDLlt27dUu4cUV5/IiKqXkziiPSsXr16yl0VvvvuO7z//vvKl/+zsrKwcuVKBAcHw9HRsUxfDw8PvPnmmwCATz75BCtWrEBWlrDAcmpqKmbPno0dO3ZotdivjY0NBg8ejOPHj0OhUAAA/vzzTwwYMABpaWmQSqVYsGCBxv6FhYUYOHAg/vzzTwDCNmLHjx/HCy+8gPz8fHh6eiqXKiEioprHJI6oGrz//vvKtdRWrVqF+vXrw8nJCY6OjliwYAGmTJmC4cOHAxCSrdJWrlyJAQMGQC6X48MPP4SjoyOcnJzQoEEDbNy4ER988AE6d+5cYQxr165FXl4eBg4cCFtbW0ilUvTs2RNXr16FtbU1du3ahcaNG2vs/+233+LevXvo2bMnpFIpbG1tMXDgQMTFxcHBwQE//fSTxmVQiIio+jGJI6oGFhYW2LNnD7Zs2YKePXtCLBajqKgI3bt3x5YtW7Bt2zZkZGQAQJntt2xsbHDo0CGsW7cOnTt3hpWVFRQKBfz9/bFnzx589tlnWsXQpEkTREVF4c0330T9+vVRUFAAV1dXTJo0CVFRUXjxxRfL7d+rVy9ERkYiMDAQ9vb2KCoqgoeHB2bNmoVr164p398jIiLD4OxUomoiEokwc+ZMzJw5s0ydQqFQLgrcrl27MvUWFhZ466238NZbb6kd+9SpU1rF4Obmhg0bNmDDhg3aB15Ky5YtERoaWqW+RIb2aO2XKHhmcpGVtzdc35lrkHiI9I1JHJEBbN++HYmJibCwsMCAAQMMHQ6RSZJduIDcq1dVysSdOhkoGiL94+NUomoyadIk7Nu3T2WG58OHD/HZZ59h1qxZAIDAwEC4u7sbKkQiIjJiTOKq2ZUrVzBv3jx06tQJdnZ2qF+/Pvr27Yv9+/dr7PPo0SPMnj0bXl5esLa2hru7O1588UXcu3ev5gInnR06dAjjx49H/fr1Ua9ePTg4OMDNzQ0LFy5EQUEB/P39sXbtWkOHSURERoqPU6vZ559/jmPHjmHMmDF44403kJubi127dmH06NFYvHhxmTW/7t27h759+8LS0hIzZsyAp6cnHj9+jEuXLiE9PR3NmjUz0Cehylq/fj0OHTqEqKgoPHr0CNnZ2ahfvz46d+6MiRMnYtq0aVwol4iIqoxJXDWbM2cOtm7dqrKMxJw5c9CnTx+sWLECc+fOVdm2aMqUKXB1dcWZM2cglZruXqV1QWBgIAIDA2v8vCVrwlWFt7e3Tv2JiKjm8HFqNfPz8yuzDpi5uTnGjBmDoqIi3L59W1l+8uRJXLx4EcuWLYNUKkVeXh7y8/NrOmQiIiIyAkziDKRkc/SSzdIB4PDhwwCEdcOee+45SCQSiMVi9OzZE2fPnjVInERERFQ7GVUSJ5PJcOjQISxfvhxjxoyBl5cXRCIRRCIRgoODtRojKysLwcHB6NChA2xtbWFvb48ePXpgzZo1KCgoqN4P8K+kpCSEhISgV69eKu+4ldyVGzt2LOzs7LB79258/fXXePDgAQYMGKB2w3QiIiKqm4zqnbhLly5h6NChVe4fFxeHfv36IfbfxR8lEgny8/MRGRmJyMhI7Ny5EydOnFC7p6VCodD60aaZmRmsrKzU1slkMowePRr5+fnYvHmzSl3JHpmtW7dGeHg4RCIRACAgIABt27bF0qVLER4eru3HJSIiIhNmVHfiAMDR0REBAQGYP38+du3aBTc3N636FRUVYfjw4YiNjYW7uzuOHTuGnJwcyGQy7N69G1KpFFFRUZg6dara/nFxcRCLxVodXbt2VTtGQUEBxowZgytXrmDnzp3o2LGjSr1YLAYgvBBfksABQIsWLdC7d2+cPn1aq89KREREps+o7sT5+/sjPT1dpWzBggVa9Q0NDcW1a9cAAGFhYfD19QUg3DWbMGEC5HI5Jk+ejIMHD+LEiRMICAhQ6e/i4oKQkBCtzqXuTl5hYSFeeuklHD16FCEhIRgzZkyZNh4eHgCgNjF1d3fHmTNnUFxcDHNzc63iICKqyxynTYN08GCVMotS7yETGTujSuJ0SV5K9n/s37+/MoErbeLEiVi0aBFiYmKwbdu2Mkmcra0tpk+fXqVzFxcXY/Lkyfjll1+wadMmBAUFqW3Xo0cPfPvtt0hMTCxTl5CQAGdnZyZwRERash/2oqFDIKpWRvc4tSpkMhnOnTsHABgyZIjaNiKRCIP//Y3t6NGjeju3XC5HUFAQ9u3bh7Vr1+K1117T2HbkyJGQSCTYvHkzCgsLleVRUVGIiIhQxkdERERkVHfiqurGjRuQy+UAgPbt22tsV1KXkpKC9PR0lUV4q2r+/PnYuXMnfH194eLigh07dqjU9+7dG02bNgUgPLL99NNPMXfuXPTt2xeTJ0/G48ePsX79ejg6OpbZ3aG0kruL169f1zlmIiIiqv3qRBJXsiYb8N97Z+qUrktOTtZLEleyLEhERAQiIiLK1IeEhCiTOAB4++234eLigjVr1mD+/PkQi8UICAjAihUrVNo968KFCzrHSkRERMajTiRxJUt3AMKyIpqUrivdRxenTp2qdJ8pU6ZgypQpejk/ERERmaY68U4cUU2bPn06RCJRlSfDEBERVaRO3IkrvZG8TCbT2K50HTefp9pi//79+Ouvv9C5c2eMGjXK0OEQEVEtUSfuxDVs2FD5dVJSksZ2petK9zEGPj4+8PHxga2traFDIT3bv38/li5div379xs6FCIiqkXqRBLXpk0bmJkJH7W82ZsldW5ubnqZ1FCTSiZOlDf7loiIiExHnUjiJBIJ/Pz8AACHDx9W20ahUODIkSMAgEGDBtVYbERERERVUSeSOADKXRJOnjyJixcvlqnfu3cv7t+/D0DYu5Souvz4448YMmQIGjRoAEtLSzg4OKBFixYYMWIENm7ciLy8PADCzGaRSKTcbSQ0NBQikUjlKD37uaioCJs3b0a/fv3g4uICS0tLODs7o1WrVpgwYQK+//57jTHt3LkTfn5+kEqlsLe3R69evbB582YoFApO0iCjlfnbATwO2apyZP52wNBhEemN0U1sePLkCYqLi5XflyziK5PJkJaWpiy3sbFReT8sKCgI69atw7Vr1zB27FiEhoYiICAAcrkcYWFhmDVrFgBhR4dnt9wi0pcZM2ao7MFra2uLwsJC3L17F3fv3sWvv/6KF198Ed7e3rCyskKDBg2QmZmJvLw82NjYwN7eXmU8KysrAMLWbkOHDsWxY8eUdfb29sjJyUF6ejpu376NPXv2YObMmSr9FQoFZs6cqYxJJBLBwcEBkZGRuHTpEk6ePAlra+vquhxE1erJ9u3IvXpVpUzcqVONbscVFZ+BP+6koU8Llxo7J9UdRpfEdenSBXFxcWXKV61ahVWrVim/DwoKwtatW5XfW1hYIDw8HP3790dsbCwGDBgAiUQCuVyuvPPRpUsX7Ny5s9o/gymInTBR67aNvt4IC2dnrdo+WLwY+bfvaNW2/ty3UU/NPrjqpG/bhqcHDpYp9/5xt1b99eGPP/5ASEgIzMzMsGLFCrzyyivKdy8fP36My5cv44cfflAmZr1790ZKSgqmT5+O0NBQTJgwQeXPdGm7du3CsWPHYGNjgw0bNmDChAmwtbWFQqFAamoqzp07hx9++KFMv6+++kqZwM2ePRtLliyBi4sLMjMz8eWXX2Lp0qVlEkciqpzpIZcQOqMn/JozkSP9MrokThfe3t6Ijo7G6tWr8dNPPyEmJgaWlpZo164dJk2ahDlz5ij/AaXyPfvbbXkUpfaBrUj+7Ttaj12cman1uIVJyZWKuTqcP38eADBgwAC8//77KnXOzs4YNGhQld/HLBk7MDBQ5W6bSCSCq6srRo8ejdGjR6v0ycvLw9KlSwEA06ZNw1dffaWss7e3x5IlS5CXl4fPPvusSjERkaBIrsCeyAQmcaR3RpfExcbG6tRfKpVi6dKlyn+8iGqKg4MDACA1NRXFxcUwNzfX+9gpKSla9zl69CjS09MBAB9//LHaNgsWLMCXX36pvFtNRJo1dpIgKj5DbV18uuY1Somqqs5MbCAytICAANjY2CAqKgr+/v74/vvvERMTo5exhw4dCpFIhPDwcAwZMgS7du1S2TNYncjISACAp6cnmjdvrraNvb09unXrppcYiUzd+G6esDATGToMqkOYxBHVkGbNmmHLli2wtbVFREQEXnnlFTRt2hSurq6YMGECfvnlFygUiiqN3adPH6xcuRJWVlY4fPgwJk+eDA8PD3h6euLll1/GyZMny/R59OgRAMDDw6PcsRs1alSlmIjqmj4tXBA6o6ehw6A6xOgep1LtIO7USeu2IktLrdtat2yhdVvzSrxwb+nRsFIxV5cpU6ZgyJAh2Lt3L06ePInz588jISEBe/bswZ49e+Dv74/ffvsNdnZ2lR57/vz5mDJlCvbs2YPTp0/j/PnzSExMxNatW7F161aMGzcOP/zwAywr8fMgosrxa+6CLo0dND5WJdInJnFUJdU1q9P9k0+qZVynwEA41ZL1/5ycnPDqq6/i1VdfBQDcu3cPW7ZswcqVK3H27FkEBwfjiy++qNLYDRs2xNy5czF37lwAwLVr17B+/Xps2bIF+/btg7+/P9566y0AgKurK4Dyt6LTpp6IiAyDj1OJDKxZs2ZYsWIFJk+eDAAqa70BUG4ZV5VHrR06dMB3332n3LGk9Njdu3cHACQkJODevXtq+z99+hSXL1+u9HmJiKj6MYkjqiH5+fnl1ovFYgD/JW0lSh6tZmRk6HXsgQMHwtHREQDwiYY7oJ9//jlyc3PLHZuIiAyDj1OJasjs2bORmZmJCRMmwN/fX/k4Mzs7Gzt27MC2bdsAAC++qLqafPv27QEAZ8+exc2bN9G6desyY48aNQru7u4YP348fH19lUuOpKen4+uvv8aJEyfKjC0Wi7F48WLMmzcPoaGhcHBwwOLFi+Hs7IynT59i3bp1+PTTT+Hg4FBuAklUW0l8fGDh5qZSZuXtbZhgiKoBkziiGlJYWIi9e/di7969AIQttywsLFQSpD59+mDRokUq/caOHYsPP/wQqampaNOmDVxcXFCvXj0AwO7du+Hj44Pc3FyEhIQod18ouXv39OlT5Tjjxo3DK6+8ojL222+/jaioKGzfvh3r1q3DV199BXt7ezx9+hTFxcWYOHEirK2tlfu3EhkT13fmGjoEomrFx6lENWTx4sVYv349Ro8ejdatW8PCwgLZ2dlwdXXFwIED8f/+3//DqVOnlAlaCUdHR5w5cwYTJ06Eh4cHMjMzERcXh7i4OOUivF999RVWrlyJoUOHokWLFlAoFMjNzUXDhg0xYsQIhIWFYe/evWUe1ZqZmWHbtm3Ytm0bfHx8IBaLUVRUhK5du+Kbb75Ru1UXERHVDrwTR1QNSpb1KK1Zs2aYM2cO5syZU+nxWrdujV27dmms79ChAzp06FBmOy9tTZs2DdOmTatSXyIiMgzeiSMiIiIyQkziiIiIiIwQkzgiIiIiI8QkjoiIiMgIcWIDEZVL3SQNIiIyPN6JIyIiIjJCTOKIiIiIjBCTOCIiIiIjxCSOiIiIyAgxiaMyFAqFoUMgE8U/W0RE+sMkjpRK9tUsLi42cCRkqkr+bD27hysREVUe/yYlJUtLS1haWiI7O9vQoZCJysrKUv45IyIi3XCdOFISiUSQSqXIyMiAvb09xGKxoUMiE5Kbm4unT5/CwcEBIpHI0OEQVZtWDaRalRHpikkcqXBxcUFubi7i4+NhZ2cHqVQKc3Nz/qNLVaJQKFBcXIysrCw8ffoU1tbWcHFxMXRYRNXqs7EdDR0C1RFM4kiFubk5PD09kZaWhqysLGRkZBg6JDIBlpaWcHBwgIuLC8zNzQ0dDhGRSWASR2WYm5ujQYMGcHV1RWFhIeRyuaFDIiNmZmYGS0tL3s0lItIzJnGkkUgkgpWVlaHDICIiIjU4O5WIiIjICDGJIyIiIjJCTOKIiIiIjBCTOCIiIiIjxCSOiIiIyAgxiSMiIiIyQkziiIiIiIyQSKFQKAwdBOlPgwYN8OjRI5UyW1tbtG/f3kARERERUUWuX7+O7OxslTJXV1c8fPhQYx8mcSZGKpWW+UNARERExsfW1hZZWVka6/k4lYiIiMgIMYkjIiIiMkJM4oiIiIiMkIWhAyD98vDwQFJSkkqZRCJB06ZN9Xqeq1evolOnTjU2hrZty2tX2TpNZebm5mXeO6zJySO6XvvK9temfUVtNNVXptzQ176m/8xr2766r33J9+peuua159831TFGXb329+/fh0wmUynz8PAov5OCqAo2bNhQo2No27a8dpWt01Tm4+OjAKBy+Pj4aBWfPuh67SvbX5v2FbXRVF+ZckNf+5r+M69t++q+9iXf89pXrg3/vqn6GLz22uPsVKJK8vX1xYULF1TKfHx8EBERYaCI6g5ee8PhtTcMXnfDMYZrzySOiIiIyAhxYgMRERGREWISR1RL3bx5E9bW1hCJRDh8+LChwzFp8fHxmDp1Klq3bg07Ozvly8vLli0rd6FN0t2VK1cwb948dOrUCXZ2dqhfvz769u2L/fv3Gzo0k5adnY3g4GAMGzYMbm5uEIlEmD59uqHDMinFxcVYuXIlmjdvDmtrazRr1gzLly9HUVGR3s7B2alEtdTrr78OS0tLFBQUGDoUk/fw4UMkJiZi9OjR8PT0hLm5OSIjI7F8+XKEh4cjIiIClpaWhg7TJH3++ec4duwYxowZgzfeeAO5ubnYtWsXRo8ejcWLF2PZsmWGDtEkpaWlYenSpXB3d0f37t1x4MABQ4dkcubMmYNNmzYhKCgIffr0wYULF7B48WLExMTg+++/189JDDuvgojU2bZtm0IsFiuWLFmiAKA4dOiQoUOqkz7//HMFAMWBAwcMHYrJ+uOPPxS5ubkqZUVFRQofHx+FhYWF4vHjxwaKzLTl5eUpEhMTFQqFQpGbm6sAoAgKCjJsUCYkOjpaIRKJFG+88YZK+bx58xQAFH/++adezsPHqUS1TEZGBt577z0sWLAA3t7ehg6nTmvSpAkA4MmTJwaOxHT5+fnBxsZGpczc3BxjxoxBUVERbt++baDITJu1tXXFa5BRle3evRsKhQJz585VKS/5fvfu3Xo5D5M4olpmwYIFsLW1xfvvv2/oUOqcvLw8pKWlISEhAQcOHMCHH34IGxsb9O3b19Ch1TnJyckAgPr16xs4EqLKi4yMhLOzM1q0aKFS7unpiYYNGyIyMlIv52ESRyZJJpPh0KFDWL58OcaMGQMvLy+IRCKIRCIEBwdrNUZWVhaCg4PRoUMH2Nrawt7eHj169MCaNWuq7T21ixcv4rvvvsO6devK3J0wFsZ67QFgy5YtqF+/Pho3boxhw4bBzMwMv/zyCzw9PavtnPpkzNe+tKSkJISEhKBXr15o1qxZjZxTF6Zy3U2NIX8uycnJGu90qttZqao4sYFM0qVLlzB06NAq94+Li0O/fv0QGxsLQNi6LD8/H5GRkYiMjMTOnTtx4sQJODo6lumrUCiQn5+v1XnMzMxgZWUFQJjJ9Prrr2Po0KEYNmxYlWM3NGO89iVGjRqF1q1bIzMzE+fPn8fp06fx9OnTKn+WmmbM176ETCbD6NGjkZ+fj82bN1f5s9QkU7jupsiQPxeZTAZnZ2e149rY2CA3N7fKcZXGO3FkshwdHREQEID58+dj165dcHNz06pfUVERhg8fjtjYWLi7u+PYsWPIycmBTCbD7t27IZVKERUVhalTp6rtHxcXB7FYrNXRtWtXZb8NGzbgxo0bWLdunV4+vyEZ27Uv0ahRIwwYMABjx47FmjVr8MEHH2D8+PE4fvy4TtejJhnrtQeAgoICjBkzBleuXMHOnTvRsWPHKl+HmmbM192UGernUpLwqZOXlwexWFzlz1Qa78SRSfL390d6erpK2YIFC7TqGxoaimvXrgEAwsLC4OvrC0D4LXbChAmQy+WYPHkyDh48iBMnTiAgIEClv4uLC0JCQrQ6V8lvcJmZmVi8eDECAwNhZmam/M0vLS0NgLAERmxsrHL5i9rM2K59ecaMGQMbGxuEhIRgwIABWo1rSMZ87QsLC/HSSy/h6NGjCAkJwZgxY7QaqzYw5utuygz5c2nYsCEuX76sduykpKQy78pVmV7muBIZAS8vLwUAxZIlS8pt5+/vrwCg6N+/v9p6uVyuaNKkiQKAIjAwUC+xxcTElNloWd2RkJCgl/PVtNp87cuTl5enMDc3VwwZMqTaz1VdjOHaFxUVKcaNG6cAoNi0aZNexzYUY7juJerSEiM19XNZuHChAoDi9u3bKuXx8fEKAIp58+ZV+TOUxsepRKXIZDKcO3cOADBkyBC1bUQiEQYPHgwAOHr0qF7O6+rqip9//rnMMWfOHADAxx9/jJ9//hkuLi56OV9tZKhrDwh3OtX59ttvUVxcjF69euntXLWRIa+9XC5HUFAQ9u3bh7Vr1+K1117T29i1nSGvO2mmj5/LhAkTIBKJ8OWXX6qUl3w/ceJEvcTKx6lEpdy4cQNyuRwA0L59e43tSupSUlKQnp4OJycnnc4rkUgwatSoMuUZGRkAAF9fX+VfGKbKUNceAD744AP8888/GDhwILy8vJCdnY3Tp0/j119/RatWrfD222/rfI7azJDXfv78+di5cyd8fX3h4uKCHTt2qNT37t0bTZs21fk8tZEhrzsgvIebkZGh3AYqOjoay5cvBwD07du3zi6to4+fS6dOnfC///0PX3/9NXJycuDv74+IiAh8//33CAoKQo8ePfQSK5M4olJK1qYCUO5CmKXrkpOT9faXal1myGs/ZswYPH78GKGhoUhNTYWFhQWaN2+Ojz76CO+99x7s7Ox0PkdtZshrX/LeUEREBCIiIsrUh4SEmGwSZ+i/b1avXo24uDjl91FRUYiKigIALFmypM4mcfr6uWzYsAFeXl7YsmULdu3ahYYNG2Lp0qVYuHCh3mJlEkdUSunNziUSicZ2peuqc4P06dOn15lNqQ157UeMGIERI0boZSxjZMhrf+rUKb2MY4wM/fdNyQQqUqWvn4uFhQUWLlyo16TtWXwnjoiIiMgIMYkjKkUqlSq/lslkGtuVrivdh6qO195weO0Ng9e9djKmnwuTOKJSGjZsqPy6vG1RSteV7kNVx2tvOLz2hsHrXjsZ08+FSRxRKW3atIGZmfC/xfXr1zW2K6lzc3PjpAY94bU3HF57w+B1r52M6efCJI6oFIlEAj8/PwDA4cOH1bZRKBQ4cuQIAGDQoEE1Fpup47U3HF57w+B1r52M6efCJI7oGUFBQQCAkydP4uLFi2Xq9+7di/v37wMAAgMDazQ2U8drbzi89obB6147Gc3PRS/7PhDVQunp6YrU1FTl4enpqQCgmD9/vkp5VlaWSr/CwkJFhw4dFAAUHh4eiuPHjysUCoWiuLhYsWfPHoWdnZ0CgFFvxVTdeO0Nh9feMHjdaydT/7kwiSOTVbJHXkWHuv0CY2JiFN7e3so2EolEYWNjo/y+S5cuivT09Jr/UEaC195weO0Ng9e9djL1nwsfpxKp4e3tjejoaHz88cdo3749RCIRLC0t0a1bN6xevRoXLlyAo6OjocM0Sbz2hsNrbxi87rWTMfxcRAqFQmHQCIiIiIio0ngnjoiIiMgIMYkjIiIiMkJM4oiIiIiMEJM4IiIiIiPEJI6IiIjICDGJIyIiIjJCTOKIiIiIjBCTOCIiIiIjxCSOiIiIyAgxiSMiIiIyQkziiIiIiIwQkzgiIgPw9vaGSCTC1q1bDR0KERkpC0MHQERkarZu3YrY2Fj069cP/fr1M3Q4RGSimMQREenZ1q1bcfr0aQDQmMQ1a9YMNjY2sLe3r8HIiMiUMIkjIjKAEydOGDoEIjJyfCeOiIiIyAgxiSMi0pOtW7dCJBIpH6UuXboUIpFI5YiNjQVQ/sSGkranTp3C48ePMW/ePDRr1gxisRheXl6YPXs2UlNTle3j4uLw+uuvo0mTJrCxsUHjxo3x7rvvIisrq9x4U1NT8dFHH6FLly6wt7eHjY0NmjZtipkzZ+Lvv//W23UhourBx6lERHoiFovRoEEDpKeno7CwEPXq1YOtra1KG3Nzc63Hi4+Px7Rp05CYmIh69epBLpcjPj4eGzduxO+//47z58/jzp07GDJkCB4/fgw7OzsUFxcjISEBX3zxBS5evIjTp0+rPefx48cxfvx4ZGRkAAAsLS1hZWWFmJgYxMTEYMeOHfjuu+8QGBio0zUhourDO3FERHoyYcIEpKSkoHfv3gCA9957DykpKSqHp6en1uO9/fbbcHFxwYULF5CdnY3s7Gzs2rULEokEN27cwOLFizF+/Hh06tQJ169fR2ZmJrKysvDVV1/B3Nwc586dQ0hISJlxr127hhEjRiAjIwOzZs3CP//8g9zcXGRnZyMuLg5vvPEGCgoKMHPmTERGRurt+hCRfjGJIyKqpaytrXH8+HH06tULgHC3bOLEiXj33XcBABs2bICtrS0OHjyIdu3aAQBsbGwwe/ZsTJ48GQCwe/fuMuPOnTsXubm5WLhwITZv3ow2bdoo79Y1btwYGzduxFtvvYWioiIsX768Jj4qEVUBkzgiolpq1qxZcHZ2LlP+wgsvKL+eN28erK2tNbaJjo5WKY+NjcXvv/8OCwsLvPfeexrPXfIY9fjx4yguLq5S/ERUvfhOHBFRLdWzZ0+15Q0aNFB+3aNHj3LbPHnyRKX83LlzAAC5XI62bdtqPHdJ4paTk4PHjx/D1dVV+8CJqEYwiSMiqqWkUqnacgsLC63bFBUVqZQnJycDEJK4hw8fahWHTCbTqh0R1SwmcUREdUjJHbYGDRogJSXFwNEQkS74ThwRUR3i5uYGAEhLS0NOTo6BoyEiXTCJIyLSMzMz4a9WhUJh4EjK8vPzAyDckTt06JCBoyEiXTCJIyLSMzs7OwBQLqRbm7Ro0QL9+vUDACxatAiZmZnltk9PT6+BqIioKpjEERHpWfv27QEABw8eRFJSkoGjKeurr76Cra0tbt++DR8fH/zyyy/Iy8tT1iclJWH79u0ICAjABx98YMBIiag8TOKIiPQsKCgINjY2uHv3Lho3bgw3Nzd4e3vD29sbiYmJhg4P7du3x+HDh+Hm5oabN29i1KhRsLW1hYuLCyQSCRo1aoTAwED8/vvvhg6ViMrB2alERHrWokULnDx5EitWrMDFixfx+PFj5VIfzy75YSh+fn64ffs2Nm/ejPDwcPz999/IyMiAWCxGmzZt0K1bNwwZMgQjR440dKhEpIFIURvfvCUiIiKicvFxKhEREZERYhJHREREZISYxBEREREZISZxREREREaISRwRERGREWISR0RERGSEmMQRERERGSEmcURERERGiEkcERERkRFiEkdERERkhJjEERERERkhJnFERERERohJHBEREZERYhJHREREZIT+P7FZdXLipCL8AAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "    \n",
    "data = run(\n",
    "    bsize=1, # batch size, number of instance to evaluate\n",
    "    n=242, # X is of shape (d,n)\n",
    "    d=242, # X is of shape (d,n)\n",
    "    p=2, # number of rows/cols to predict, A is of shape (d-p,n-p), D is of shape (p,p)\n",
    "    r=240, # rank of X\n",
    "    s=240, # subsampling of cols in A, set s = n-p to deactivate subsampling\n",
    "    m=1, # number of machines in distributed setting, set m = 1 to run centralized\n",
    "    eps=0, # standard deviation of additive data noise\n",
    "    alpha=.5, # controls data diversity in distribute setting, \n",
    "             # note that this does NOT directly translate to the diversity coefficient alpha in the paper\n",
    "             # set to alpha = 1 for very easy distributed problem, alpha = 0.5 for difficult problem\n",
    "    cond=2, # log-condition number of A\n",
    "    prec=1e-10, # target precision, iterative solvers stop when the prediction changes by less than this in an iteration\n",
    "    dat=\"orthogonal\", # data sampling, only \"orthogonal\" allows to control the condition number \n",
    ")\n",
    "\n",
    "hue_order=[\"gdpp\", \"gd\", \"cg\", \"lstsq\"]\n",
    "colors = sns.color_palette(\"tab10\", n_colors=len(hue_order))\n",
    "palette = dict(zip(hue_order, colors))\n",
    "ax = sns.lineplot(\n",
    "    data=data.groupby([\"iteration\", \"solver\"]).mean().reset_index(), \n",
    "    x=\"time\", # or \"iteration\"\n",
    "    y=\"error\", \n",
    "    hue=\"solver\", \n",
    "    drawstyle=\"steps-post\", \n",
    "    style=\"solver\", \n",
    "    dashes={\"lstsq\": (2, 2), 'gd': \"\", 'cg': \"\", 'gdpp': \"\"},\n",
    "    palette=palette,\n",
    ")\n",
    "\n",
    "plt.xscale(\"log\")\n",
    "plt.yscale(\"log\")\n",
    "plt.show()\n",
    "    "
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "dac",
   "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.20"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
