Units

In synthesizer all quantities a user interacts with (that are not dimensionless) have units associated with them, implemented with the unyt package.

Synthesizer objects and methods should always be provided with quantites and associated units. This can be easily achieved with the unyt package.

[1]:
from unyt import Mpc

# Define a variable with units
x = 1 * Mpc

print(x)
print("x is now a unyt_quantity: type(x)=", type(x))
1 Mpc
x is now a unyt_quantity: type(x)= <class 'unyt.array.unyt_quantity'>

All unit functionality in synthesizer is contained in the units module. In this module we have defined a dictionary containing the default units of all attributes throughout synthesizer.

[2]:
from synthesizer.units import default_units

print(default_units)
{'lam': Å, 'obslam': Å, 'wavelength': Å, 'vacuum_wavelength': Å, 'original_lam': Å, 'lam_min': Å, 'lam_max': Å, 'lam_eff': Å, 'lam_fwhm': Å, 'mean_lams': Å, 'pivot_lams': Å, 'nu': Hz, 'obsnu': Hz, 'nuz': Hz, 'original_nu': Hz, 'luminosity': erg/s, 'luminosities': erg/s, 'bolometric_luminosity': erg/s, 'bolometric_luminosities': erg/s, 'lnu': erg/(Hz*s), 'llam': erg/(s*Å), 'continuum': erg/(Hz*s), 'flux': erg/(cm**2*s), 'fnu': nJy, 'flam': erg/(cm**2*s*Å), 'equivalent_width': Å, 'coordinates': Mpc, 'radii': Mpc, 'smoothing_lengths': Mpc, 'softening_length': Mpc, 'velocities': km/s, 'mass': unyt_quantity(1., 'Msun'), 'masses': unyt_quantity(1., 'Msun'), 'initial_masses': unyt_quantity(1., 'Msun'), 'initial_mass': unyt_quantity(1., 'Msun'), 'current_masses': unyt_quantity(1., 'Msun'), 'dust_masses': unyt_quantity(1., 'Msun'), 'ages': yr, 'accretion_rate': unyt_quantity(1., 'Msun/yr'), 'accretion_rates': unyt_quantity(1., 'Msun/yr'), 'bb_temperature': K, 'bb_temperatures': K, 'inclination': degree, 'inclinations': degree, 'resolution': Mpc, 'fov': Mpc, 'orig_resolution': Mpc, 'centre': Mpc, 'photo_lnu': erg/(Hz*s), 'photo_fnu': erg/(Hz*cm**2*s), 'softening_lengths': Mpc}

As you can see, most units are defined symbollically, however, you may notice units such as Msun (the default used for masses) is defined as a compound unit in terms of kgs. Shortly we will cover working with the quantities returned by synthesizer and how compound units work in practice.

The Units object

The unit system is defined by the Units object. This object contains a collection of attributes defining the units associated to each quantity throughout synthesizer which is not dimensionless. Importantly, Units is a Singleton object. This means there can only ever be one instance of Units; if a second is instantiated then the first is returned. This ensures that the unit system remains consistent when running synthesizer.

[3]:
from synthesizer.units import Units

# Define multiple Units instances
units1 = Units()
units2 = Units()

print("Both units instances are the same object:", units1 is units2)
Both units instances are the same object: True

You can take a look at the unit system by printing the instance of Units.

[4]:
print(units1)
Unit System:
lam:                Å
obslam:             Å
wavelength:         Å
vacuum_wavelength:  Å
original_lam:       Å
lam_min:            Å
lam_max:            Å
lam_eff:            Å
lam_fwhm:           Å
mean_lams:          Å
pivot_lams:         Å
nu:                 Hz
obsnu:              Hz
nuz:                Hz
original_nu:        Hz
luminosity:         erg/s
luminosities:       erg/s
bolometric_luminosity: erg/s
bolometric_luminosities: erg/s
lnu:                erg/(Hz*s)
llam:               erg/(s*Å)
continuum:          erg/(Hz*s)
flux:               erg/(cm**2*s)
fnu:                nJy
flam:               erg/(cm**2*s*Å)
equivalent_width:   Å
coordinates:        Mpc
radii:              Mpc
smoothing_lengths:  Mpc
softening_length:   Mpc
velocities:         km/s
mass:               0.9999999999999999 Msun
masses:             0.9999999999999999 Msun
initial_masses:     0.9999999999999999 Msun
initial_mass:       0.9999999999999999 Msun
current_masses:     0.9999999999999999 Msun
dust_masses:        0.9999999999999999 Msun
ages:               yr
accretion_rate:     0.9999999999999999 Msun/yr
accretion_rates:    0.9999999999999999 Msun/yr
bb_temperature:     K
bb_temperatures:    K
inclination:        degree
inclinations:       degree
resolution:         Mpc
fov:                Mpc
orig_resolution:    Mpc
centre:             Mpc
photo_lnu:          erg/(Hz*s)
photo_fnu:          erg/(Hz*cm**2*s)
softening_lengths:  Mpc

Modifying the default unit system

If the default unit system works for your needs then you don’t need to do anything. You will never interact with the Units object and all quantites will have the default units associated to them automatically. However, if you need to change one or more of the units used you can import Units and instantiate it with a dictionary of the modified quantities.

[5]:
from unyt import Myr, kpc, Msun

# Make the dictionary containing the units we want to change
new_units = {
    "coordinates": kpc,
    "smoothing_lengths": kpc,
    "softening_length": kpc,
    "ages": Myr,
}

# Set up the modified unit system
units = Units(new_units)

print()
print(units)

Unit System:
lam:                Å
obslam:             Å
wavelength:         Å
vacuum_wavelength:  Å
original_lam:       Å
lam_min:            Å
lam_max:            Å
lam_eff:            Å
lam_fwhm:           Å
mean_lams:          Å
pivot_lams:         Å
nu:                 Hz
obsnu:              Hz
nuz:                Hz
original_nu:        Hz
luminosity:         erg/s
luminosities:       erg/s
bolometric_luminosity: erg/s
bolometric_luminosities: erg/s
lnu:                erg/(Hz*s)
llam:               erg/(s*Å)
continuum:          erg/(Hz*s)
flux:               erg/(cm**2*s)
fnu:                nJy
flam:               erg/(cm**2*s*Å)
equivalent_width:   Å
coordinates:        Mpc
radii:              Mpc
smoothing_lengths:  Mpc
softening_length:   Mpc
velocities:         km/s
mass:               0.9999999999999999 Msun
masses:             0.9999999999999999 Msun
initial_masses:     0.9999999999999999 Msun
initial_mass:       0.9999999999999999 Msun
current_masses:     0.9999999999999999 Msun
dust_masses:        0.9999999999999999 Msun
ages:               yr
accretion_rate:     0.9999999999999999 Msun/yr
accretion_rates:    0.9999999999999999 Msun/yr
bb_temperature:     K
bb_temperatures:    K
inclination:        degree
inclinations:       degree
resolution:         Mpc
fov:                Mpc
orig_resolution:    Mpc
centre:             Mpc
photo_lnu:          erg/(Hz*s)
photo_fnu:          erg/(Hz*cm**2*s)
softening_lengths:  Mpc

Something has gone wrong… but recall that the unit system will return the original if one exists, so actually this should be completely expected.

This issue highlights the need to set up Units before doing anything else. If any computations have been done the Units instance will exist and will not be modifiable after the fact. However, should you fall in this trap the code will warn you as above - no hidden gotchas here!

Now, lets go against the advice above, and use the highly inadvisable force argument to get a new Unit system. But please note, in a real use case, forcing a modified unit system WILL NOT convert existing quantities to the new unit system.

[6]:
# Set up the modified unit system
units = Units(new_units, force=True)

print()
print(units)
Redefining unit system:
coordinates: kpc
smoothing_lengths: kpc
softening_length: kpc
ages: Myr

Unit System:
lam:                Å
obslam:             Å
wavelength:         Å
vacuum_wavelength:  Å
original_lam:       Å
lam_min:            Å
lam_max:            Å
lam_eff:            Å
lam_fwhm:           Å
mean_lams:          Å
pivot_lams:         Å
nu:                 Hz
obsnu:              Hz
nuz:                Hz
original_nu:        Hz
luminosity:         erg/s
luminosities:       erg/s
bolometric_luminosity: erg/s
bolometric_luminosities: erg/s
lnu:                erg/(Hz*s)
llam:               erg/(s*Å)
continuum:          erg/(Hz*s)
flux:               erg/(cm**2*s)
fnu:                nJy
flam:               erg/(cm**2*s*Å)
equivalent_width:   Å
coordinates:        kpc
radii:              Mpc
smoothing_lengths:  kpc
softening_length:   kpc
velocities:         km/s
mass:               0.9999999999999999 Msun
masses:             0.9999999999999999 Msun
initial_masses:     0.9999999999999999 Msun
initial_mass:       0.9999999999999999 Msun
current_masses:     0.9999999999999999 Msun
dust_masses:        0.9999999999999999 Msun
ages:               Myr
accretion_rate:     0.9999999999999999 Msun/yr
accretion_rates:    0.9999999999999999 Msun/yr
bb_temperature:     K
bb_temperatures:    K
inclination:        degree
inclinations:       degree
resolution:         Mpc
fov:                Mpc
orig_resolution:    Mpc
centre:             Mpc
photo_lnu:          erg/(Hz*s)
photo_fnu:          erg/(Hz*cm**2*s)
softening_lengths:  Mpc

Working with Quantity objects

There is no need to work with the Units object itself beyond initially defining a modified unit system. Beyond this, all unit operations are handled “behind the scenes”. This hidden functionality is enabled by the Quantity object.

All attributes on synthesizer objects which carry units are in fact Quantity objects. Quantitiy objects carry a reference to the global unit system, and extract the appropriate units depending on the name of the variable storing the Quantity. As such, a user will never instantiate a quantity themselves, but their usage is important.

One simple thing to keep in mind is how to return the value with or without units. This is achieved by the application or omission of a leading underscore to a variable name.

Lets create an Sed object, which has a wavelength array stored under lam.

[7]:
import numpy as np
from unyt import Hz, angstrom, erg, s

from synthesizer.sed import Sed

# Make an sed with arbitrary arguments
sed = Sed(
    lam=np.linspace(10, 1000, 10) * angstrom,
    lnu=np.ones(10) * erg / s /Hz
)

We can access this attribute with units as you would expect to access any attribute.

[8]:
print(sed.lam)
[  10.  120.  230.  340.  450.  560.  670.  780.  890. 1000.] Å

Or we can append a leading underscore and return it without units.

[9]:
print(sed._lam)
[  10.  120.  230.  340.  450.  560.  670.  780.  890. 1000.]

In the case of compound units this is somewhat less elegant. Lets demonstrate with a Stars object.

[10]:
from synthesizer.particle.stars import Stars

# Create a dummy Stars object
stars = Stars(
    initial_masses=np.random.rand(10) * Msun,
    ages=np.ones(10) * Myr,
    metallicities=np.ones(10),
)

If we print the initial_masses with units we get the compound version in kg.

[11]:
print(stars.initial_masses)
[0.80700882 0.34693926 0.42397661 0.26977888 0.73707274 0.76545786
 0.84844152 0.02054982 0.02425346 0.75854512] Msun

However, if we extract the values alone we get the values we expect in \(M_\odot\).

[12]:
print(stars._initial_masses)
[0.80700882 0.34693926 0.42397661 0.26977888 0.73707274 0.76545786
 0.84844152 0.02054982 0.02425346 0.75854512]

Its worth keeping this in mind whenever extracting masses from synthesizer objects.

Automatic unit conversion

Finally, let’s utilise some automatic unit conversion. If we input a mixture of properties to a synthesizer object, all with different units to the global unit system, we don’t have to convert them all before inputting them. As long as we pass them to synthesizer with unyt units attached, the conversion will be handled automatically. Here we use a Stars object again to demonstrate.

[13]:
from unyt import Mpc, g, m

# Create a dummy Stars object
stars = Stars(
    initial_masses=np.random.rand(10) * 10**34.0 * g,
    ages=np.ones(10) * Myr,
    metallicities=np.ones(10),
    coordinates=np.random.rand(10, 3) * Mpc,
    smoothing_lengths=np.random.rand(10) * 10**22.0 * m,
)

print(
    "stars.initial_masses[0]=",
    stars.initial_masses[0],
    "=",
    stars._initial_masses[0],
    "Msun",
)
print("stars.ages[0]=", stars.ages[0])
print("stars.coordinates[0]=", stars.coordinates[0])
print("stars.smoothing_lengths[0]=", stars.smoothing_lengths[0])
stars.initial_masses[0]= 4.395714312068567 Msun = 4.395714312068568 Msun
stars.ages[0]= 1000000.0 yr
stars.coordinates[0]= [0.69363104 0.55775963 0.44476959] Mpc
stars.smoothing_lengths[0]= 0.2836032456210852 Mpc