gcmotion.QuantityConstructor#
A light reading of pint’s Defining Quantities paragraph is recommended. Note that we use the notation Q instead of Q_.
Initializing the Quantity Constructor#
Since we like to use Normalized Units a lot, the QuantityConstructor()
function extends pint’s build-in units to include NU. Since the normalization is done with respect to the tokamak’s major radius R, the magnetic field strength B0 on the magnetic axis and on the particle’s mass (which is defined through its species), these three values must be the first to be defined. To make our lives easier, we also define the minor radius a and pass it to to the QuantityConstructor, who will also create psi_wall as a unit of magnetic flux. This means that a quantity of 1psi_wall is equal to the magnetic flux on the tokamak’s last closed surface. That way we can define initial \(\psi_0\) s with respect to the tokamak’s wall, instead of guessing!
Here are some examples of how the Quantity Constructor Q is used in gcmotion:
>>> import gcmotion as gcm
>>> import numpy as np
>>>
>>> # Define R, a, B0 and 'species' and lastly the Quantity Constructor 'Q'
>>> Rnum = 1.65 # meters
>>> anum = 0.5 # meters
>>> B0num = 1 # Tesla
>>> species = "D"
>>> Q = gcm.QuantityConstructor(R=Rnum, a=anum, B0=B0num, species=species)
>>>
>>> # Defining R, a and B0 correctly
>>> R = Q(Rnum, "meters")
>>> a = Q(anum, "meters")
>>> B0 = Q(B0num, "Tesla")
Now R, a and B0 are defined as Quantities, and can be used to create every sort of configuration. Mind you that their initial numeric values Rnum, anum and B0num must represent meters for the Constructor to define the units correctly, and you should reuse their values to define the Quantities, to avoid any errors. However, you could just as well do:
>>> R = Q(1650, "millimeters")
or
>>> R = Q(1.744051376e-16, "lightyears")
Tip
All classes make sure that their parameters are instanciated in the correct units internally, no matter how you defined them, as long as they have the correct dimensionality.
Calling the constructor#
Here is the Constructor’s documentation.
- gcmotion.QuantityConstructor(*args, **kwargs)#
Defining complex Quantities#
Sometimes we must define Quantities with units more complex than “meters” or “Tesla”. For example, to define the strength of an electric field we can do:
>>> Ea = Q(100, "kiloVolts/meter")
>>> Ea
<Quantity(100, 'kilovolt / meter')>
>>> print(Ea)
100 kilovolt / meter
To define the toroidal and poloidal currents, we use the
>>> i = Q(0, "Plasma_current")
>>> g = Q(0.5, "Plasma_current")
Here, “Plasma_current” is an alias to “Tesla * meters”, which is the unit of the currents in SI. However, we would rather define g = 1 in NU. We will see how to do that in the next chapter.
To define a Quantity with units of Magnetic_flux, we would do the following:
>>> psi = Q(5, "Magnetic_flux")
>>> print(psi)
5 Magnetic_flux
>>> print(psi.to("Tesla * meters^2"))
5.0 meter ** 2 * tesla
>>> print(psi.to_base_units())
5.0 kilogram * meter ** 2 / ampere / second ** 2
Here, “Magnetic_flux” is but an alias to “Tesla * meter^2”
We can convert “Magnetic_flux” to “psi_wall”. That essentially shows us psi’s position relative to the wall:
>>> print(psi_wall.to("psi_wall"))
40.0 psi_wall
With that logic, we can easily define initial conditions relative to the device’s wall:
>>> psi0 = Q(0.85, "psi_wall")
>>> print(psi0.to("Tesla * meters ^ 2"))
0.10625 meter ** 2 * tesla
Tip
Pint’s string parser is very powerful: Not only it understands operations between units without needing a very strict syntax, it can also perform operations between Quantities and resolve the resulting dimentionality. Moreover, it offers a method of typechecking, since it doesn’t allow for illegal operations (try adding R and B0!). However, avoid using abbreviations like ‘mm’ or ‘kg’, since ‘kg’ actually stands for ‘kilogauss’ instead of ‘kilogram’. Finally, the constructor is case-insensitive and does not distinguish between pluralized forms (e.g. ‘meter’ and ‘meters’.)
Converting to NU and back#
You can also convert values to their corresponding NU units. For example:
>>> R = Q(Rnum, "meters")
>>> print(R)
1.65 meter
>>> print(round(R.to("NUmeters")))
1.0 NUmeter
>>> print(round(B0.to("NUTesla")))
1.0 NUtesla
but
>>> print(a.to("NUmeters"))
0.30303030303030304 NUmeter
This is a good sanity check to make sure the normalizations are defined correctly
To define the toroidal and poloidal currents in NU, we can similarly:
>>> i = Q(0, "NUPlasma_current")
>>> g = Q(1, "NUPlasma_current")
>>> print(g)
1 NUPlasma_current
>>> print(g.to("Tesla*meter"))
1.65 meter * tesla
Conversion Table#
Here is a list of the availiable NU units:
Normalised Unit |
Can be converted to: |
Description |
---|---|---|
NUsecond |
second, hour, nanosecond, … |
Normalized time unit |
NUw0 |
Hz, Mhz, 1/s, … |
Normalized frequency unit |
NUmeter |
meter, kilometer, micrometer, … |
Normalized length unit |
NUJoule |
Joule, eV, keV, … |
Normalized energy unit |
NUTesla |
Tesla |
Normalized magnetic flux density unit |
NUvelocity |
m/s, km/h, … |
Normalized velocity unit |
NUMagnetic_flux |
Tesla * meter^2 |
Normalized magnetic flux unit |
NUCanonical_momentum |
Canonical_momentum, Joule * second |
Normalized Canonical Momentum |
NUPlasma_current |
Tesla * meter |
Normalized plasma current unit |
NUMagnetic_moment |
Ampere * meter^2 |
Normalized magnetic moment unit |
NUVolts |
Volts |
Normalized electric potential strength unit |
NUVolts_per_NUmeter |
Volts/meter |
Normalized electric field strength unit |
NUpsi_wall |
psi_wall, Tesla * meter^2 |
Normalized psi_wall |