"""
Classes and methods to interface with files storing rate data.
"""
import re
from pathlib import Path
from pynucastro.constants import constants
from pynucastro.nucdata.elements import PeriodicTable, UnidentifiedElement
from pynucastro.nucdata.halflife_table import HalfLifeTable
from pynucastro.nucdata.mass_table import MassTable
from pynucastro.nucdata.partition_function import PartitionFunctionCollection
from pynucastro.nucdata.spin_table import SpinTable
_pynucastro_dir = Path(__file__).parents[1]
_pynucastro_rates_dir = _pynucastro_dir/'library'
_pynucastro_tabular_dir = _pynucastro_rates_dir/'tabular'
# read the various tables with nuclear properties at the module-level
_mass_table = MassTable()
_halflife_table = HalfLifeTable()
_spin_table = SpinTable(reliable=True)
# read the partition function table once and store it at the module-level
_pcollection = PartitionFunctionCollection(use_high_temperatures=True, use_set='frdm')
[docs]
class UnsupportedNucleus(Exception):
pass
[docs]
class Nucleus:
"""
a nucleus that participates in a reaction -- we store it in a
class to hold its properties, define a sorting, and give it a
pretty printing string.
:var Z: atomic number
:var N: neutron number
:var A: atomic mass
:var nucbind: nuclear binding energy (MeV / nucleon)
:var short_spec_name: nucleus abbreviation (e.g. "he4")
:var caps_name: capitalized short species name (e.g. "He4")
:var el: element name (e.g. "he")
:var pretty: LaTeX formatted version of the nucleus name
:var dm: mass excess (MeV)
:var A_nuc: nuclear mass (amu)
:var mass: nuclear mass (MeV)
:var tau: half life (s)
"""
_cache = {}
def __init__(self, name, dummy=False):
name = name.lower()
self.raw = name
# a dummy nucleus is one that we can use where a nucleus is needed
# but it is not considered to be part of the network
self.dummy = dummy
self.nse = False
# element symbol and atomic weight
if name == "p":
self.el = "h"
self.A = 1
self.short_spec_name = "h1"
self.caps_name = "p"
elif name == "d":
self.el = "h"
self.A = 2
self.short_spec_name = "h2"
self.caps_name = "H2"
elif name == "t":
self.el = "h"
self.A = 3
self.short_spec_name = "h3"
self.caps_name = "H3"
elif name == "a":
#this is a convenience, enabling the use of a commonly-used alias:
# He4 --> \alpha --> "a" , e.g. c12(a,g)o16
self.el = "he"
self.A = 4
self.short_spec_name = "he4"
self.raw = "he4"
self.caps_name = "He4"
elif name == "n":
self.el = "n"
self.A = 1
self.Z = 0
self.N = 1
self.short_spec_name = "n"
self.spec_name = "neutron"
self.pretty = fr"\mathrm{{{self.el}}}"
self.caps_name = "n"
elif name == "p_nse":
# this is a proton with a different name
# it is meant to be used in iron-group rates
# in NSE
self.el = "h"
self.A = 1
self.Z = 1
self.N = 0
self.short_spec_name = "p_nse"
self.spec_name = "proton-nse"
self.pretty = r"\mathrm{p}_\mathrm{NSE}"
self.caps_name = "p_NSE"
self.nse = True
elif name.strip() in ("al-6", "al*6"):
raise UnsupportedNucleus()
else:
e = re.match(r"([a-zA-Z]*)(\d*)", name)
self.el = e.group(1).title() # chemical symbol
assert self.el
self.A = int(e.group(2))
assert self.A >= 0
self.short_spec_name = name
self.caps_name = name.capitalize()
# use lowercase element abbreviation regardless the case of the input
self.el = self.el.lower()
# atomic number comes from periodic table
if name not in ["n", "p_nse"]:
i = PeriodicTable.lookup_abbreviation(self.el)
self.Z = i.Z
assert isinstance(self.Z, int)
assert self.Z >= 0
self.N = self.A - self.Z
assert isinstance(self.N, int)
assert self.N >= 0
# long name
self.spec_name = f'{i.name}-{self.A}'
# latex formatted style
self.pretty = fr"{{}}^{{{self.A}}}\mathrm{{{self.el.capitalize()}}}"
# set the number of spin states
try:
self.spin_states = _spin_table.get_spin_states(a=self.A, z=self.Z)
except NotImplementedError:
self.spin_states = None
# set a partition function object to every nucleus
try:
self.partition_function = _pcollection.get_partition_function(self.short_spec_name)
except ValueError:
self.partition_function = None
# nuclear mass
try:
# Note: for Nubase 2020, we need to use the CODATA 18 constants
mass_H = _mass_table.get_mass_diff(a=1, z=1) + constants.m_u_MeV_C18
self.dm = _mass_table.get_mass_diff(a=self.A, z=self.Z)
self.A_nuc = float(self.A) + self.dm / constants.m_u_MeV_C18
self.mass = self.A * constants.m_u_MeV_C18 + self.dm
B = (self.Z * mass_H + self.N * constants.m_n_MeV_C18) - self.mass
self.nucbind = B / self.A
except NotImplementedError:
self.dm = None
self.A_nuc = None
self.mass = None
self.nucbind = None
# halflife
try:
self.tau = _halflife_table.get_halflife(a=self.A, z=self.Z)
except NotImplementedError:
self.tau = None
[docs]
@classmethod
def from_cache(cls, name, dummy=False):
key = (name.lower(), dummy)
if key not in cls._cache:
cls._cache[key] = cls(name, dummy)
return cls._cache[key]
[docs]
@classmethod
def from_Z_A(cls, Z, A, dummy=False):
"""creates a nucleus given Z and A"""
# checks if Z and A are valid inputs
if not (isinstance(Z, int) and isinstance(A, int)):
raise TypeError("Nucleus Z and A must be integers")
if not (Z >= 0 and A >= 0):
raise ValueError("Nucleus Z and A must be non-negative")
if Z > A:
raise ValueError("Nucleus Z can't be bigger than A")
# checks if neutron
if (Z, A) == (0, 1):
return cls.from_cache("n", dummy)
# otherwise, finds element Z on the periodic table
i = PeriodicTable.lookup_Z(Z)
if i is None:
raise UnidentifiedElement(f"Element {Z} could not be found")
name = i.abbreviation + str(A)
return cls.from_cache(name, dummy)
def __repr__(self):
if self.raw not in ("p", "p_nse", "d", "t", "n"):
return self.raw.capitalize()
return self.raw
def __hash__(self):
return hash((self.Z, self.A))
[docs]
def c(self):
"""return the capitalized-style name"""
return self.caps_name
[docs]
def cindex(self):
"""return the name for C++ indexing"""
return self.short_spec_name.capitalize()
def __eq__(self, other):
if isinstance(other, Nucleus):
return (self.el == other.el and
self.Z == other.Z and self.A == other.A and
self.nse == other.nse)
if isinstance(other, tuple):
return (self.Z, self.A) == other
return NotImplemented
def __lt__(self, other):
if not self.Z == other.Z:
return self.Z < other.Z
return self.A < other.A
def __add__(self, other):
Z = self.Z + other.Z
A = self.A + other.A
dummy = self.dummy and other.dummy
return Nucleus.from_Z_A(Z, A, dummy)
def __sub__(self, other):
Z = self.Z - other.Z
A = self.A - other.A
dummy = self.dummy and other.dummy
return Nucleus.from_Z_A(Z, A, dummy)
[docs]
@classmethod
def cast(cls, obj):
if isinstance(obj, cls):
return obj
if isinstance(obj, str):
return cls.from_cache(obj)
raise TypeError("Invalid type passed to Nucleus.cast() (expected str or Nucleus)")
[docs]
@classmethod
def cast_list(cls, lst, *, allow_None=False, allow_single=False):
if allow_None and lst is None:
return lst
if isinstance(lst, (str, cls)):
if allow_single:
return [cls.cast(lst)]
raise ValueError("Single object passed to Nucleus.cast_list() instead of list")
return [cls.cast(obj) for obj in lst]
[docs]
def get_nuclei_in_range(zmin, zmax, amin, amax):
"""given a range of Z = [zmin, zmax], and A = [amin, amax],
return a list of Nucleus objects for all nuclei in this range"""
nuc_list = []
assert zmax >= zmin, "zmax must be >= zmin"
assert amax >= amin, "amax must be >= amin"
for z in range(zmin, zmax+1):
element = PeriodicTable.lookup_Z(z)
for a in range(amin, amax+1):
name = f"{element.abbreviation}{a}"
nuc_list.append(Nucleus(name))
return nuc_list
[docs]
def get_all_nuclei():
"""Return a list will every Nucleus that has a known mass"""
nuc_list = []
for (A, Z) in _mass_table.mass_diff:
if Z == 0 and A == 1:
nuc = "n"
else:
el = PeriodicTable.lookup_Z(Z)
nuc = f"{el.abbreviation}{A}"
nuc_list.append(Nucleus(nuc))
return nuc_list