{ "cells": [ { "cell_type": "markdown", "metadata": { "id": "DS4E1PagbDgL" }, "source": [ "# Setup" ] }, { "cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": [ "!echo \"Runtime started.\"" ] }, { "cell_type": "markdown", "metadata": { "id": "tQ1r1bbb0yBv" }, "source": [ "\n", "## Instructions\n", "1. Work on a copy of this notebook: _File_ > _Save a copy in Drive_ (you will need a Google account).\n", "2. (Optional) If you would like to do the deep learning component of this tutorial, ensure the GPU accelerator is turned on, with Edit->Notebook settings->Hardware accelerator->GPU\n", "3. **Use fallback runtime**. Run the above cell (`!echo \"Runtime started.\"`) to start the runtime. Now, open the command pallette (bottom left -> second icon from the bottom), and search for \"use fallback runtime\", and hit enter.\n", " - This is a temporary workaround for a bug in Colab, until the current runtime is patched.\n", "4. Execute the following cell (click on it and press Ctrl+Enter or Shift+Enter) to install Julia, IJulia and other packages. This takes a couple of minutes.\n", "5. Continue to the next section.\n", "\n", "_Notes_:\n", "* If your Colab Runtime gets reset (e.g., due to inactivity), repeat steps 4-5.\n", "* After installation, if you want to change the Julia version or activate/deactivate the GPU, you will need to reset the Runtime: _Runtime_ > _Delete and disconnect runtime_ and repeat steps 2-5.\n", "\n", "> **Warning**\n", "> \n", "> Ensure that you have done step 3 above, otherwise you will get an error when you try to use Julia." ] }, { "cell_type": "markdown", "metadata": { "id": "COndi88gbDgO" }, "source": [ "**Run the following code to install Julia**" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "GIeFXS0F0zww" }, "outputs": [], "source": [ "%%shell\n", "set -e\n", "\n", "#---------------------------------------------------#\n", "JULIA_VERSION=\"1.7.2\"\n", "JULIA_NUM_THREADS=4\n", "export JULIA_PKG_PRECOMPILE_AUTO=0\n", "#---------------------------------------------------#\n", "\n", "if [ -z `which julia` ]; then\n", " # Install Julia\n", " JULIA_VER=`cut -d '.' -f -2 <<< \"$JULIA_VERSION\"`\n", " echo \"Installing Julia $JULIA_VERSION on the current Colab Runtime...\"\n", " BASE_URL=\"https://julialang-s3.julialang.org/bin/linux/x64\"\n", " URL=\"$BASE_URL/$JULIA_VER/julia-$JULIA_VERSION-linux-x86_64.tar.gz\"\n", " wget -nv $URL -O /tmp/julia.tar.gz # -nv means \"not verbose\"\n", " tar -x -f /tmp/julia.tar.gz -C /usr/local --strip-components 1\n", " rm /tmp/julia.tar.gz\n", "\n", " echo \"Installing PyCall.jl...\"\n", " julia -e 'using Pkg; Pkg.add(\"PyCall\"); Pkg.build(\"PyCall\")'\n", " julia -e 'println(\"Success\")'\n", "\n", "fi" ] }, { "cell_type": "markdown", "metadata": { "id": "ORv1c6xvbDgV" }, "source": [ "Install PySR and PyTorch-Lightning:" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "EhMRSZEYFPLz" }, "outputs": [], "source": [ "%pip install -Uq pysr pytorch_lightning" ] }, { "cell_type": "markdown", "metadata": { "id": "etTMEV0wDqld" }, "source": [ "The following step is not normally required, but colab's printing is non-standard and we need to manually set it up PyJulia:\n" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "j666aOI8xWF_" }, "outputs": [], "source": [ "from julia import Julia\n", "\n", "julia = Julia(compiled_modules=False)\n", "from julia import Main\n", "from julia.tools import redirect_output_streams\n", "\n", "redirect_output_streams()\n" ] }, { "cell_type": "markdown", "metadata": { "id": "6u2WhbVhht-G" }, "source": [ "Let's install the backend of PySR, and all required libraries.\n", "\n", "**(This may take some time)**" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "J-0QbxyK1_51" }, "outputs": [], "source": [ "import pysr\n", "\n", "pysr.install()\n" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "vFpyRxmhFqeH" }, "outputs": [], "source": [ "import sympy\n", "import numpy as np\n", "from matplotlib import pyplot as plt\n", "from pysr import PySRRegressor\n", "import torch\n", "from torch import nn, optim\n", "from torch.nn import functional as F\n", "from torch.utils.data import DataLoader, TensorDataset\n", "import pytorch_lightning as pl\n", "from sklearn.model_selection import train_test_split\n" ] }, { "cell_type": "markdown", "metadata": { "id": "gsRMQ7grbDga" }, "source": [ "# Simple PySR example:\n" ] }, { "cell_type": "markdown", "metadata": { "id": "myTEwdiUFiGL" }, "source": [ "First, let's learn a simple function\n", "\n", "$$2.5382 \\cos(x3) + x0^2 - 2$$" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "Cb1eb2XuFQh8" }, "outputs": [], "source": [ "# Dataset\n", "np.random.seed(0)\n", "X = 2 * np.random.randn(100, 5)\n", "y = 2.5382 * np.cos(X[:, 3]) + X[:, 0] ** 2 - 2\n" ] }, { "cell_type": "markdown", "metadata": { "id": "cturCkaVjzLs" }, "source": [ "By default, we will set up 30 populations of expressions (which evolve independently except for migrations), use 4 threads, and use `\"best\"` for our model selection strategy:" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "4nDAAnisdhTc" }, "outputs": [], "source": [ "default_pysr_params = dict(\n", " populations=30,\n", " procs=4,\n", " model_selection=\"best\",\n", ")\n" ] }, { "cell_type": "markdown", "metadata": { "id": "N4gANfkaj8ie" }, "source": [ "PySR can run for arbitrarily long, and continue to find more and more accurate expressions. You can set the total number of cycles of evolution with `niterations`, although there are also a [few more ways](https://github.com/MilesCranmer/PySR/pull/134) to stop execution.\n", "\n", "**This first execution will take a bit longer to startup, as the library is JIT-compiled. The next execution will be much faster.**" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "p4PSrO-NK1Wa", "scrolled": true }, "outputs": [], "source": [ "# Learn equations\n", "model = PySRRegressor(\n", " niterations=30,\n", " binary_operators=[\"plus\", \"mult\"],\n", " unary_operators=[\"cos\", \"exp\", \"sin\"],\n", " **default_pysr_params\n", ")\n", "\n", "model.fit(X, y)\n" ] }, { "cell_type": "markdown", "metadata": { "id": "-bsAECbdkQsQ" }, "source": [ "We can print the model, which will print out all the discovered expressions:" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "4HR8gknlZz4W" }, "outputs": [], "source": [ "model\n" ] }, { "cell_type": "markdown", "metadata": { "id": "ME3ddPxXkWQg" }, "source": [ "We can also view the SymPy format of the best expression:" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "IQKOohdpztS7" }, "outputs": [], "source": [ "model.sympy()\n" ] }, { "cell_type": "markdown", "metadata": { "id": "EHIIPlmClltn" }, "source": [ "We can also view the SymPy of any other expression in the list, using the index of it in `model.equations_`." ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "GRcxq-TTlpRX" }, "outputs": [], "source": [ "model.sympy(2)\n" ] }, { "cell_type": "markdown", "metadata": { "id": "YMugcGX4tbqj" }, "source": [ "## Output" ] }, { "cell_type": "markdown", "metadata": { "id": "gIWt5wz5cjXE" }, "source": [ "`model.equations_` is a Pandas DataFrame. We can export the results in various ways:" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "HFGaNL6tbDgi" }, "outputs": [], "source": [ "model.latex()\n" ] }, { "cell_type": "markdown", "metadata": { "id": "4hS8kqutcmPQ" }, "source": [ "These is also `model.sympy(), model.jax(), model.pytorch()`. All of these can take an index as input, to get the result for an arbitrary equation in the list.\n", "\n", "We can also use `model.predict` for arbitrary equations, with the default equation being the one chosen by `model_selection`:" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "Vbz4IMsk2NYH" }, "outputs": [], "source": [ "ypredict = model.predict(X)\n", "ypredict_simpler = model.predict(X, 2)\n", "\n", "print(\"Default selection MSE:\", np.power(ypredict - y, 2).mean())\n", "print(\"Manual selection MSE for index 2:\", np.power(ypredict_simpler - y, 2).mean())\n" ] }, { "cell_type": "markdown", "metadata": { "id": "SQDUScGebDgr" }, "source": [ "# Custom operators" ] }, { "cell_type": "markdown", "metadata": { "id": "qvgVbOoSFtQY" }, "source": [ "A full list of operators is given here: https://astroautomata.com/PySR/operators,\n", "but we can also use any binary or unary operator in `julia`, or define our own as arbitrary functions.\n", "\n", "Say that we want a command to do quartic powers:\n", "\n", "$$ y = x_0^4 - 2 $$" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "JvXOVqSyFsdr" }, "outputs": [], "source": [ "y = X[:, 0] ** 4 - 2\n" ] }, { "cell_type": "markdown", "metadata": { "id": "-zoqaL8KGSK5" }, "source": [ "We can do this by passing a string in Julia syntax.\n", "\n", "We also define the operator in sympy, with `extra_sympy_mappings`, to enable its use in `predict`, and other export functions." ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "PoEkpvYuGUdy", "scrolled": true }, "outputs": [], "source": [ "model = PySRRegressor(\n", " niterations=5,\n", " populations=40,\n", " binary_operators=[\"plus\", \"mult\"],\n", " unary_operators=[\"cos\", \"exp\", \"sin\", \"quart(x) = x^4\"],\n", " extra_sympy_mappings={\"quart\": lambda x: x**4},\n", ")\n", "model.fit(X, y)\n" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "emn2IajKbDgy" }, "outputs": [], "source": [ "model.sympy()\n" ] }, { "cell_type": "markdown", "metadata": { "id": "wbWHyOjl2_kX" }, "source": [ "Since `quart` is arguably more complex than the other operators, you can also give it a different complexity, using, e.g., `complexity_of_operators={\"quart\": 2}` to give it a complexity of 2 (instead of the default 2). You can also define custom complexities for variables and constants (`complexity_of_variables` and `complexity_of_constants`, respectively - both take a single number).\n", "\n", "\n", "One can also add a binary operator, with, e.g., `\"myoperator(x, y) = x^2 * y\"`. All Julia operators that work on scalar 32-bit floating point values are available.\n", "\n", "Make sure that any operator you add is valid over the real line. So, e.g., you will need to define `\"mysqrt(x) = sqrt(abs(x))\"` to enable it for negative numbers,\n", "or, simply have it return a very large number for bad inputs (to prevent negative input in a soft way):\n", "`\"mysqrt(x::T) where {T} = (x >= 0) ? x : T(-1e9)\"` (Julia syntax for a template function of input type `T`), which will make `mysqrt(x)` return -10^9 for negative x–hurting the loss of the equation." ] }, { "cell_type": "markdown", "metadata": { "id": "pEXT4xskbDg0" }, "source": [ "## Scoring" ] }, { "cell_type": "markdown", "metadata": { "id": "IyeYbVVOG60w" }, "source": [ "Using `model_selection=\"best\"`selects the equation with the max score and prints it. But in practice it is best to look through all the equations manually, select an equation above some MSE threshold, and then use the score to select among that loss threshold.\n", "\n", "Here, \"score\" is defined by:\n", "$$ \\text{score} = - \\log(\\text{loss}_i/\\text{loss}_{i-1})/\n", "(\\text{complexity}_i - \\text{complexity}_{i-1})$$" ] }, { "cell_type": "markdown", "metadata": { "id": "I3IxmvSQrhfw" }, "source": [ "This scoring is motivated by the common strategy of looking for drops in the loss-complexity curve.\n", "\n", "From Schmidt & Lipson (2009) -" ] }, { "cell_type": "markdown", "metadata": { "id": "eUeXyoLxrd8o" }, "source": [ "![F4.large.jpg]()" ] }, { "cell_type": "markdown", "metadata": { "id": "gDZyxsA7bDg9" }, "source": [ "# Noise example" ] }, { "cell_type": "markdown", "metadata": { "id": "cJCHdDt6IOou" }, "source": [ "Here is an example with noise. Known Gaussian noise with $\\sigma$ between 0.1 and 5.0. We record samples of $y$:\n", "\n", "$$ \\sigma \\sim U(0.1, 5.0) $$\n", "$$ \\epsilon \\sim \\mathcal{N}(0, \\sigma^2)$$\n", "$$ y = 5\\;\\cos(3.5 x_0) - 1.3 + \\epsilon.$$\n", "We have 5 features, say. The weights change the loss function to be:\n", "$$MSE = \\sum [(y - f(x))^2*w],$$\n", "\n", "so in this example, we can set:\n", "$$w = 1/\\sigma^2.$$" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "up1RvmwyOdal" }, "outputs": [], "source": [ "np.random.seed(0)\n", "N = 3000\n", "upper_sigma = 5\n", "X = 2 * np.random.rand(N, 5)\n", "sigma = np.random.rand(N) * (5 - 0.1) + 0.1\n", "eps = sigma * np.random.randn(N)\n", "y = 5 * np.cos(3.5 * X[:, 0]) - 1.3 + eps\n" ] }, { "cell_type": "markdown", "metadata": { "id": "-EJPDZbP5YEZ" }, "source": [ "Let's look at this dataset:" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "sqMqb4nJ5ZR5" }, "outputs": [], "source": [ "plt.scatter(X[:, 0], y, alpha=0.2)\n", "plt.xlabel(\"$x_0$\")\n", "plt.ylabel(\"$y$\")\n" ] }, { "cell_type": "markdown", "metadata": { "id": "kaddasbBuDDv" }, "source": [ "Define some weights to use:" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "3wqz9_sIbDhA" }, "outputs": [], "source": [ "weights = 1 / sigma ** 2\n" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "v8WBYtcZbDhC" }, "outputs": [], "source": [ "weights[:5]\n" ] }, { "cell_type": "markdown", "metadata": { "id": "NXWdQSCFuAzV" }, "source": [ "Let's run PySR again:" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "a07K3KUjOxcp", "scrolled": true }, "outputs": [], "source": [ "model = PySRRegressor(\n", " loss=\"myloss(x, y, w) = w * abs(x - y)\", # Custom loss function with weights.\n", " niterations=20,\n", " populations=20, # Use more populations\n", " binary_operators=[\"plus\", \"mult\"],\n", " unary_operators=[\"cos\"],\n", ")\n", "model.fit(X, y, weights=weights)\n" ] }, { "cell_type": "markdown", "metadata": { "id": "CHCMO9CouFLP" }, "source": [ "Let's see if we get similar results to the true equation" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "oHyUbcg6ggmx" }, "outputs": [], "source": [ "model\n" ] }, { "cell_type": "markdown", "metadata": { "id": "OchlZZQP8Ums" }, "source": [ "We can also filter all equations up to 2x the most accurate equation, then select the best score from that list:" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "PB67POLr8b_L" }, "outputs": [], "source": [ "best_idx = model.equations_.query(\n", " f\"loss < {2 * model.equations_.loss.min()}\"\n", ").score.idxmax()\n", "model.sympy(best_idx)\n" ] }, { "cell_type": "markdown", "metadata": { "id": "SRHTP4x55roh" }, "source": [ "We can also use `denoise=True`, which will run the input through a Gaussian process to denoise the dataset, before fitting on it." ] }, { "cell_type": "markdown", "metadata": { "id": "eTGQ4NA78yAw" }, "source": [ "Let's look at the fit:" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "ezCC0IkS8zFf" }, "outputs": [], "source": [ "plt.scatter(X[:, 0], y, alpha=0.1)\n", "y_prediction = model.predict(X, index=best_idx)\n", "plt.scatter(X[:, 0], y_prediction)\n" ] }, { "cell_type": "markdown", "metadata": { "id": "cPc1EDvRbDhL" }, "source": [ "# High-dimensional input: Neural Nets + Symbolic Regression" ] }, { "cell_type": "markdown", "metadata": { "id": "3hS2kTAbbDhL" }, "source": [ "In this example, let's learn a high-dimensional problem. **This will use the method proposed in our NeurIPS paper: https://arxiv.org/abs/2006.11287.**\n", "\n", "Let's consider a time series problem:\n", "\n", "$$ z = y^2,\\quad y = \\frac{1}{100} \\sum(y_i),\\quad y_i = x_{i0}^2 + 6 \\cos(2*x_{i2})$$\n", "\n", "Imagine our time series is 100 timesteps. That is very hard for symbolic regression, even if we impose the inductive bias of $$z=f(\\sum g(x_i))$$ - it is the square of the number of possible equations!\n", "\n", "But, as in our paper, **we can break this problem down into parts with a neural network. Then approximate the neural network with the symbolic regression!**\n", "\n", "Then, instead of, say, $(10^9)^2=10^{18}$ equations, we only have to consider $2\\times 10^9$ equations." ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "SXJGXySlbDhL" }, "outputs": [], "source": [ "###### np.random.seed(0)\n", "N = 100000\n", "Nt = 100\n", "X = 6 * np.random.rand(N, Nt, 5) - 3\n", "y_i = X[..., 0] ** 2 + 6 * np.cos(2 * X[..., 2])\n", "y = np.sum(y_i, axis=1) / y_i.shape[1]\n", "z = y**2\n", "X.shape, y.shape\n" ] }, { "cell_type": "markdown", "metadata": { "id": "8ZqGupq_uSgp" }, "source": [ "## Neural Network definition" ] }, { "cell_type": "markdown", "metadata": { "id": "r2NR0h8-bDhN" }, "source": [ "So, as described above, let's first use a neural network with the sum inductive bias to solve this problem.\n", "\n", "Essentially, we will learn two neural networks:\n", "- `f`\n", "- `g`\n", "\n", "each defined as a multi-layer perceptron. We will sum over `g` the same way as in our equation, but we won't define the summed part beforehand.\n", "\n", "Then, we will fit `g` and `f` **separately** using symbolic regression." ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "nWVfkV_YbDhO" }, "outputs": [], "source": [ "hidden = 128\n", "total_steps = 50000\n", "\n", "\n", "def mlp(size_in, size_out, act=nn.ReLU):\n", " return nn.Sequential(\n", " nn.Linear(size_in, hidden),\n", " act(),\n", " nn.Linear(hidden, hidden),\n", " act(),\n", " nn.Linear(hidden, hidden),\n", " act(),\n", " nn.Linear(hidden, size_out),\n", " )\n", "\n", "\n", "class SumNet(pl.LightningModule):\n", " def __init__(self):\n", " super().__init__()\n", "\n", " ########################################################\n", " # The same inductive bias as above!\n", " self.g = mlp(5, 1)\n", " self.f = mlp(1, 1)\n", "\n", " def forward(self, x):\n", " y_i = self.g(x)[:, :, 0]\n", " y = torch.sum(y_i, dim=1, keepdim=True) / y_i.shape[1]\n", " z = self.f(y)\n", " return z[:, 0]\n", "\n", " ########################################################\n", "\n", " # PyTorch Lightning bookkeeping:\n", " def training_step(self, batch, batch_idx):\n", " x, z = batch\n", " predicted_z = self(x)\n", " loss = F.mse_loss(predicted_z, z)\n", " return loss\n", "\n", " def validation_step(self, batch, batch_idx):\n", " return self.training_step(batch, batch_idx)\n", "\n", " def configure_optimizers(self):\n", " self.trainer.reset_train_dataloader()\n", "\n", " optimizer = torch.optim.Adam(self.parameters(), lr=self.max_lr)\n", " scheduler = {\n", " \"scheduler\": torch.optim.lr_scheduler.OneCycleLR(\n", " optimizer,\n", " max_lr=self.max_lr,\n", " total_steps=self.total_steps,\n", " final_div_factor=1e4,\n", " ),\n", " \"interval\": \"step\",\n", " }\n", " return [optimizer], [scheduler]\n" ] }, { "cell_type": "markdown", "metadata": { "id": "kK725aSEuUvG" }, "source": [ "## Data bookkeeping" ] }, { "cell_type": "markdown", "metadata": { "id": "KdWVtWUcbDhQ" }, "source": [ "Put everything into PyTorch and do a train/test split:" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "0ym19abgbDhR" }, "outputs": [], "source": [ "Xt = torch.tensor(X).float()\n", "zt = torch.tensor(z).float()\n", "X_train, X_test, z_train, z_test = train_test_split(Xt, zt, random_state=0)\n", "train_set = TensorDataset(X_train, z_train)\n", "train = DataLoader(train_set, batch_size=128, num_workers=2)\n", "test_set = TensorDataset(X_test, z_test)\n", "test = DataLoader(test_set, batch_size=256, num_workers=2)\n" ] }, { "cell_type": "markdown", "metadata": { "id": "3dw_NefuudIq" }, "source": [ "## Train the model with PyTorch Lightning on GPUs:" ] }, { "cell_type": "markdown", "metadata": { "id": "hhlhLQUBbDhT" }, "source": [ "Start the model:" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "1ldN0999bDhU" }, "outputs": [], "source": [ "pl.seed_everything(0)\n", "model = SumNet()\n", "model.total_steps = total_steps\n", "model.max_lr = 1e-2\n" ] }, { "cell_type": "markdown", "metadata": { "id": "WWRsu5A9bDhW" }, "source": [ "PyTorch Lightning trainer object:" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "33R2nrv-b62w" }, "outputs": [], "source": [ "trainer = pl.Trainer(max_steps=total_steps, gpus=1, benchmark=True)\n" ] }, { "cell_type": "markdown", "metadata": { "id": "jh91CukM5CkI" }, "source": [ "Here, we fit the neural network:" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "TXZdF8k1bDhY" }, "outputs": [], "source": [ "trainer.fit(model, train_dataloaders=train, val_dataloaders=test)\n" ] }, { "cell_type": "markdown", "metadata": { "id": "uYzk0yU4ulfH" }, "source": [ "## Latent vectors of network\n", "\n", "Let's get the input and output of the learned `g` function from the network over some random data:" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "s2sQLla5bDhb" }, "outputs": [], "source": [ "np.random.seed(0)\n", "idx = np.random.randint(0, 10000, size=1000)\n", "\n", "X_for_pysr = Xt[idx]\n", "y_i_for_pysr = model.g(X_for_pysr)[:, :, 0]\n", "y_for_pysr = torch.sum(y_i_for_pysr, dim=1) / y_i_for_pysr.shape[1]\n", "z_for_pysr = zt[idx] # Use true values.\n", "\n", "X_for_pysr.shape, y_i_for_pysr.shape\n" ] }, { "cell_type": "markdown", "metadata": { "id": "nCCIvvAGuyFi" }, "source": [ "## Learning over the network:\n", "\n", "Now, let's fit `g` using PySR:" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "51QdHVSkbDhc", "scrolled": true }, "outputs": [], "source": [ "np.random.seed(1)\n", "tmpX = X_for_pysr.detach().numpy().reshape(-1, 5)\n", "tmpy = y_i_for_pysr.detach().numpy().reshape(-1)\n", "idx2 = np.random.randint(0, tmpy.shape[0], size=3000)\n", "\n", "model = PySRRegressor(\n", " niterations=20,\n", " binary_operators=[\"plus\", \"sub\", \"mult\"],\n", " unary_operators=[\"cos\", \"square\", \"neg\"],\n", ")\n", "model.fit(X=tmpX[idx2], y=tmpy[idx2])\n" ] }, { "cell_type": "markdown", "metadata": { "id": "xginVMmTu3MZ" }, "source": [ "## Validation" ] }, { "cell_type": "markdown", "metadata": { "id": "6WuaeqyqbDhe" }, "source": [ "Recall we are searching for $y_i$ above:\n", "\n", "$$ z = y^2,\\quad y = \\frac{1}{100} \\sum(y_i),\\quad y_i = x_{i0}^2 + 6 \\cos(2 x_{i2})$$" ] }, { "cell_type": "code", "execution_count": null, "metadata": { "id": "E1_VWQ45bDhf" }, "outputs": [], "source": [ "model\n" ] }, { "cell_type": "markdown", "metadata": { "id": "mlU1hidZkgCY" }, "source": [ "A neural network can easily undo a linear transform, so this is fine: the network for $f$ will learn to undo the linear transform.\n", "\n", "Then, we can learn another analytic equation for $z$." ] }, { "cell_type": "markdown", "metadata": { "id": "TntGlQEwbDhk" }, "source": [ "**Now, we can compose these together to get the time series model!**\n", "\n", "Think about what we just did: we found an analytical equation for $z$ in terms of $500$ datapoints, under the assumption that $z$ is a function of a sum of another function over an axis:\n", "\n", "$$ z = f(\\sum_i g(x_i)) $$\n", "\n", "And we pulled out analytical copies for $g$ using symbolic regression." ] }, { "cell_type": "markdown", "metadata": { "id": "1QsHVjAVbDhk" }, "source": [ "# Other PySR Options" ] }, { "cell_type": "markdown", "metadata": { "id": "S5dO61g1bDhk" }, "source": [ "The full list of PySR parameters can be found here: https://astroautomata.com/PySR/api" ] } ], "metadata": { "accelerator": "GPU", "colab": { "name": "pysr_demo.ipynb", "provenance": [] }, "language_info": { "name": "python" }, "kernelspec": { "name": "python3", "display_name": "Python 3" }, "gpuClass": "standard" }, "nbformat": 4, "nbformat_minor": 0 }