{ "cells": [ { "cell_type": "markdown", "metadata": {}, "source": [ "## Tutorial on how to 'delay' the start of particle advection" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "In many applications, it is needed to 'delay' the start of particle advection. For example because particles need to be released at different times throughout an experiment. Or because particles need to be released at a constant rate from the same set of locations.\n", "\n", "This tutorial will show how this can be done. We start with importing the relevant modules." ] }, { "cell_type": "code", "execution_count": 1, "metadata": {}, "outputs": [], "source": [ "%matplotlib inline\n", "from parcels import FieldSet, ParticleSet, JITParticle, Variable, ParticleFile, plotTrajectoriesFile\n", "from parcels import AdvectionRK4\n", "import numpy as np\n", "import xarray as xr\n", "from datetime import timedelta as delta\n", "import matplotlib.pyplot as plt" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "First import a `FieldSet` (from the Peninsula example, in this case)" ] }, { "cell_type": "code", "execution_count": 2, "metadata": {}, "outputs": [], "source": [ "fieldset = FieldSet.from_parcels('Peninsula_data/peninsula', allow_time_extrapolation = True)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "Now, there are two ways to delay the start of particles. Either by defining the whole `ParticleSet` at initialisation and giving each particle its own `time`. Or by using the `repeatdt` argument. We will show both options here" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "### Assigning each particle its own `time` ###" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "The simplest way to delaye the start of a particle is to use the `time` argument for each particle" ] }, { "cell_type": "code", "execution_count": 3, "metadata": {}, "outputs": [], "source": [ "npart = 10 # number of particles to be released\n", "lon = 3e3 * np.ones(npart)\n", "lat = np.linspace(3e3 , 45e3, npart, dtype=np.float32)\n", "time = np.arange(0, npart) * delta(hours=1).total_seconds() # release every particle one hour later\n", "\n", "pset = ParticleSet(fieldset=fieldset, pclass=JITParticle, lon=lon, lat=lat, time=time)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "Then we can execute the `pset` as usual" ] }, { "cell_type": "code", "execution_count": 4, "metadata": {}, "outputs": [ { "name": "stderr", "output_type": "stream", "text": [ "INFO: Compiled ArrayJITParticleAdvectionRK4 ==> /var/folders/r2/8593q8z93kd7t4j9kbb_f7p00000gr/T/parcels-504/lib15ab640308c954e580b6f93be742baae_0.so\n" ] } ], "source": [ "output_file = pset.ParticleFile(name=\"DelayParticle_time.zarr\", outputdt=delta(hours=1))\n", "pset.execute(AdvectionRK4, runtime=delta(hours=24), dt=delta(minutes=5),\n", " output_file=output_file)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "And then finally, we can show a movie of the particles. Note that the southern-most particles start to move first." ] }, { "cell_type": "code", "execution_count": 5, "metadata": {}, "outputs": [ { "data": { "text/html": [ "" ], "text/plain": [ "" ] }, "execution_count": 5, "metadata": {}, "output_type": "execute_result" } ], "source": [ "plotTrajectoriesFile('DelayParticle_time.zarr', mode='movie2d_notebook')" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "### Using the `repeatdt` argument ###" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "The second method to delay the start of particle releases is to use the `repeatdt` argument when constructing a `ParticleSet`. This is especially useful if you want to repeatedly release particles from the same set of locations." ] }, { "cell_type": "code", "execution_count": 6, "metadata": {}, "outputs": [], "source": [ "npart = 10 # number of particles to be released\n", "lon = 3e3 * np.ones(npart)\n", "lat = np.linspace(3e3 , 45e3, npart, dtype=np.float32)\n", "repeatdt = delta(hours=3) # release from the same set of locations every 3 hours\n", "\n", "pset = ParticleSet(fieldset=fieldset, pclass=JITParticle, lon=lon, lat=lat, repeatdt=repeatdt)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "Now we again define an output file and execute the `pset` as usual." ] }, { "cell_type": "code", "execution_count": 7, "metadata": {}, "outputs": [ { "name": "stderr", "output_type": "stream", "text": [ "INFO: Compiled ArrayJITParticleAdvectionRK4 ==> /var/folders/r2/8593q8z93kd7t4j9kbb_f7p00000gr/T/parcels-504/libcbcec39cf73cfa7671e8798abaac733b_0.so\n" ] } ], "source": [ "output_file = pset.ParticleFile(name=\"DelayParticle_releasedt\", outputdt=delta(hours=1))\n", "pset.execute(AdvectionRK4, runtime=delta(hours=24), dt=delta(minutes=5),\n", " output_file=output_file)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "And we get an animation where a new particle is released every 3 hours from each start location" ] }, { "cell_type": "code", "execution_count": 8, "metadata": {}, "outputs": [ { "data": { "text/html": [ "" ], "text/plain": [ "" ] }, "execution_count": 8, "metadata": {}, "output_type": "execute_result" } ], "source": [ "plotTrajectoriesFile('DelayParticle_releasedt.zarr', mode='movie2d_notebook')" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "Note that, if you want to if you want to at some point stop the repeatdt, the easiest implementation is to use two calls to `pset.execute()`. For example, if in the above example you only want four releases of the pset, you could do the following" ] }, { "cell_type": "code", "execution_count": 9, "metadata": {}, "outputs": [ { "name": "stderr", "output_type": "stream", "text": [ "INFO: Compiled ArrayJITParticleAdvectionRK4 ==> /var/folders/r2/8593q8z93kd7t4j9kbb_f7p00000gr/T/parcels-504/libd1c17d6554e6d39a5c989beb50bcc2d3_0.so\n" ] }, { "data": { "text/html": [ "" ], "text/plain": [ "" ] }, "execution_count": 9, "metadata": {}, "output_type": "execute_result" } ], "source": [ "pset = ParticleSet(fieldset=fieldset, pclass=JITParticle, lon=lon, lat=lat, repeatdt=repeatdt)\n", "output_file = pset.ParticleFile(name=\"DelayParticle_releasedt_9hrs\", outputdt=delta(hours=1))\n", "\n", "# first run for 3 * 3 hrs\n", "pset.execute(AdvectionRK4, runtime=delta(hours=9), dt=delta(minutes=5),\n", " output_file=output_file)\n", "\n", "# now stop the repeated release\n", "pset.repeatdt = None\n", "\n", "# now continue running for the remaining 15 hours\n", "pset.execute(AdvectionRK4, runtime=delta(hours=15), dt=delta(minutes=5),\n", " output_file=output_file)\n", "\n", "plotTrajectoriesFile('DelayParticle_releasedt_9hrs.zarr', mode='movie2d_notebook')" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "### Synced `time` in the output file\n", "\n", "Note that, because the `outputdt` variable controls the JIT-loop, all particles are written _at the same time_, even when they start at a non-multiple of `outputdt`. \n", "\n", "For example, if your particles start at `time=[0, 1, 2]` and `outputdt=2`, then the times written (for `dt=1` and `endtime=4`) will be " ] }, { "cell_type": "code", "execution_count": 10, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "[[ 0 2 4]\n", " [ 2 4 'NaT']\n", " [ 2 4 'NaT']]\n" ] } ], "source": [ "outtime_expected = np.array([[0, 2, 4], [2, 4, np.datetime64(\"NaT\")], [2, 4, np.datetime64(\"NaT\")]], dtype=\"timedelta64[s]\")\n", "print(outtime_expected)" ] }, { "cell_type": "code", "execution_count": 11, "metadata": {}, "outputs": [ { "name": "stderr", "output_type": "stream", "text": [ "INFO: Compiled ArrayJITParticleAdvectionRK4 ==> /var/folders/r2/8593q8z93kd7t4j9kbb_f7p00000gr/T/parcels-504/lib73f1058452aa339434e56df780d0ba46_0.so\n" ] } ], "source": [ "outfilepath = \"DelayParticle_nonmatchingtime.zarr\"\n", "\n", "pset = ParticleSet(fieldset=fieldset, pclass=JITParticle,\n", " lat=[3e3]*3, lon=[3e3]*3, time=[0, 1, 2])\n", "pset.execute(AdvectionRK4, endtime=4, dt=1, output_file=pset.ParticleFile(name=outfilepath, outputdt=2))" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "And indeed, the `time` values in the NetCDF output file are as expected" ] }, { "cell_type": "code", "execution_count": 12, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "[[ 0 2 4]\n", " [ 2 4 'NaT']\n", " [ 2 4 'NaT']]\n" ] } ], "source": [ "outtime_infile = xr.open_dataset(outfilepath).time.values[:]\n", "print(outtime_infile.astype('timedelta64[s]'))\n", "\n", "assert (outtime_expected[np.isfinite(outtime_expected)] == outtime_infile[np.isfinite(outtime_infile)]).all()" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "Now, for some applications, this behavior may be undesirable; for example when particles need to be analyzed at a same age (instead of at a same time). In that case, we recommend either changing `outputdt` so that it is a common divisor of all start times; or doing multiple Parcels runs with subsets of the original `ParticleSet` (e.g., in the example above, one run with the Particles that start at `time=[0, 2]` and one with the Particle at `time=[1]`). In that case, you will get two files:" ] }, { "cell_type": "code", "execution_count": 13, "metadata": {}, "outputs": [ { "name": "stderr", "output_type": "stream", "text": [ "INFO: Compiled ArrayJITParticleAdvectionRK4 ==> /var/folders/r2/8593q8z93kd7t4j9kbb_f7p00000gr/T/parcels-504/lib4a9283c7ab81279666f072b74de9d644_0.so\n" ] }, { "name": "stdout", "output_type": "stream", "text": [ "[[ 0 2 4]\n", " [ 2 4 'NaT']]\n" ] }, { "name": "stderr", "output_type": "stream", "text": [ "INFO: Compiled ArrayJITParticleAdvectionRK4 ==> /var/folders/r2/8593q8z93kd7t4j9kbb_f7p00000gr/T/parcels-504/libef5ef37165aa5febcc2e916a89504269_0.so\n" ] }, { "name": "stdout", "output_type": "stream", "text": [ "[[1 3 4]]\n" ] } ], "source": [ "for times in [[0,2], [1]]:\n", " pset = ParticleSet(fieldset=fieldset, pclass=JITParticle,\n", " lat=[3e3]*len(times), lon=[3e3]*len(times), time=times)\n", " pset.execute(AdvectionRK4, endtime=4, dt=1, output_file=pset.ParticleFile(name=outfilepath, outputdt=2))\n", " print(xr.open_dataset(outfilepath).time.values[:].astype('timedelta64[s]'))" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "### Adding new particles to a ParticleSet during runtime" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "In the examples above, all particles were defined at the start of the simulation. There are use-cases, though, where it is important to be able to add particles 'on-the-fly', during the runtime of a Parcels simulation.\n", "\n", "Unfortuantely, Parcels does not (yet) support adding new particles _within_ a `Kernel`. A workaround is to temporarily leave the `execution()` call to add particles via the `ParticleSet.add()` method, before continuing with `execution()`.\n", "\n", "See the example below, where we add 'mass' to a particle each timestep, based on a probablistic condition, and then split the particle once its 'mass' is larger than 5" ] }, { "cell_type": "code", "execution_count": 14, "metadata": {}, "outputs": [ { "name": "stderr", "output_type": "stream", "text": [ "INFO: Compiled ArrayGrowingParticleGrowParticles ==> /var/folders/s5/gxtkk3c12yqgd7hkt1b_s0vr0000gq/T/parcels-503/libb16839740d01629200df57019f843d0e_0.so\n" ] } ], "source": [ "class GrowingParticle(JITParticle):\n", " mass = Variable('mass', initial=0)\n", " splittime = Variable('splittime', initial=-1)\n", " splitmass = Variable('splitmass', initial=0)\n", "\n", "def GrowParticles(particle, fieldset, time):\n", " if ParcelsRandom.random() < 0.25: # 25% chance per timestep for particle to grow\n", " particle.mass += 1.\n", " if (particle.mass >= 5.) and (particle.splittime < 0):\n", " particle.splittime = time\n", " particle.splitmass = particle.mass / 2.\n", " particle.mass = particle.mass / 2.\n", "\n", "pset = ParticleSet(fieldset=fieldset, pclass=GrowingParticle, lon=0, lat=0)\n", "outfile = ParticleFile('growingparticles.zarr', pset, outputdt=1)\n", "\n", "for t in range(40):\n", " pset.execute(GrowParticles, runtime=1, dt=1, output_file=outfile)\n", " for p in pset:\n", " if p.splittime > 0:\n", " pset.add(ParticleSet(fieldset=fieldset, pclass=GrowingParticle, lon=0, lat=0, \n", " time=p.splittime, mass=p.splitmass))\n", " p.splittime = -1 # reset splittime" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "The 'trick' is that we place the `pset.execute()` call in a for-loop, so that we leave the JIT-mode and can add Particles to the ParticleSet. \n", "\n", "Indeed, if we plot the mass of particles as a function of time, we see that new particles are created every time a particle reaches a mass of 5." ] }, { "cell_type": "code", "execution_count": 15, "metadata": {}, "outputs": [ { "data": { "image/png": "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\n", "text/plain": [ "
" ] }, "metadata": { "needs_background": "light" }, "output_type": "display_data" } ], "source": [ "ds = xr.open_zarr('growingparticles.zarr')\n", "plt.plot(ds.time.values[:].astype('timedelta64[s]').T, ds.mass.T)\n", "plt.grid()\n", "plt.xlabel('Time')\n", "plt.ylabel(\"Particle 'mass'\")\n", "plt.show()\n", "ds.close()" ] } ], "metadata": { "kernelspec": { "display_name": "Python 3 (ipykernel)", "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.10" } }, "nbformat": 4, "nbformat_minor": 1 }