Common Pitfalls¶
I get errors like “Unknown units” or “Not convertible” from analysis or plotting routines¶
One of the great things about pynbody is that it takes care of units, but if it can’t figure out what units to use, everything is assumed to be dimensionless. Some analysis functions then get grumpy, because they’re trying to be smart about using sensible units but the information just isn’t available. The simplest way to avoid this situation is to make sure pynbody can work out the units for itself.
In particular, for gasoline/PKD/tipsy users, make sure you have a param file in the directory where you are analyzing a tipsy file, and make sure that it defines dKpcUnit and dMsolUnit.
Even if you are analyzing a DM only simulation, it’s can be easier to play along and assign units even though they weren’t needed for the simulation.
I tried to make an image but got a generic-looking error spewed back at me¶
The most common causes of uninformative errors in the image-making process are that:
there are no particles in the view you tried to generate; or
all the particles are clustered in the central pixel.
To tackle both of these issues in turn:
1. The image is always centred on
(0,0,0)
, so you need to offset the simulation before you start. The most common way to do this is with the functionpynbody.analysis.halo.center()
; see Basic snapshot manipulation for an introduction.2. The width keyword for the image function expects a floating point number in the current units of the snapshot. It also defaults to the number 10, which may be very large compared to your snapshot, depending on the units you have adopted. That means either you should specify a width which is more appropriate (i.e. your call might look like
pynbody.plot.sph.image(my_snapshot, width=0.01)
) or convert to sensible units first. The easiest way to do the latter is to callphysical_units()
on your snapshot, e.g.s.physical_units()
if your simulation is calleds
.
Note
Point 2. above should be fixed by commit a11df5af1f10.
I’m trying to load a file which I’m sure is fine, but it says the format isn’t recognized¶
When you load a file with pynbody you don’t have to specify what
format it is. This is quite handy when everything works. However, if
there is a minor problem with the file, pynbody may move onto trying
to interpret it as a different format and ultimately conclude it just
doesn’t understand what’s going on. At that point you’ll get an unhelpful error
like this: File 'filename': format not understood or does not exist
.
To expose the underlying problem, you need to explicitly tell pynbody
what file format you think it is. For instance, if you have a tipsy
file, try replacing your pynbody.load(filename)
with
pynbody.snapshot.tipsy.TipsySnap(filename)
. This will then show you the
actual error. Most likely it’s to do with file permissions, or a
problem with a .param
file. If at this point you can’t see what’s
going wrong, do drop us a line.
I’m using a RAMSES snapshot, but when I try to open it I get bizarre errors about “No space left on device” or something similar¶
This will happen if you are using the parallel loader of RAMSES snapshots and your shared memory has filled up. Look in your /dev/shm for files named “pynbody-*” – there are probably many of them. If your code exits gracefully, these files are deleted, but if your session crashes or you kill your python shell for whatever reason, these files will stick around. You should periodically delete them by hand if they start to accumulate.
Note that the amount of shared memory available (which is used by the parallel loader) to processes on your machine is normally substantially less than the total amount of memory on your machine. You can tell the kernel to allow more; see here, for instance.
Alternatively, you can disable parallel loading on ramses. Create a
.pynbodyrc
file in your home directory with the following
section:
[ramses]
parallel-read: 0
If you are still having trouble, contact us.
When processing multiple files I run out of memory, even if I process them one-by-one¶
Pynbody lets you have as many files in memory as you can fit, since each is stored in a self-contained objection. Clearly, if you try to load too many at once, you’re going to run out of memory.
However sometimes you might intend to load files one-by-one and still find the memory usage is cumulative.
for i in outputs:
s = pynbody.load(i)
do_something_with(s)
If this happens to you, the cause is almost certainly that python is not garbage-collecting the snapshot.
You can force python to tidy up by using its gc module:
import gc
for i in outputs:
s = pynbody.load(i)
do_something_with(s)
del s
gc.collect()
If you still have problems you may have extra references to the
snapshot. In python, del
only deletes a reference to an object,
not the object itself. You need to del
every reference, or let the
reference fall out of scope, before the garbage collector will do
anything.
import gc
s = pynbody.load("my_file")
s2 = s
del s
gc.collect() # does nothing
del s2
gc.collect() # success
Note that a SubSnap
holds a reference to its parent
SimSnap
. Any references to a SubSnap
will keep the parent
SimSnap
alive. On the other hand, a SimArray
holds only weak
references, so it won’t keep a SimSnap
alive.
import gc
s = pynbody.load("my_file")
s2 = s.dm
ar = s['mass']
del s
gc.collect() # does nothing
del s2
gc.collect() # success, ar alone is left in memory