43#include "tkdatabase.h"
58using json = nlohmann::json;
60tkmanager* tkmanager::g_data_manager =
nullptr;
64 if(g_data_manager)
return g_data_manager;
66 std::lock_guard<std::recursive_mutex> lock(get_mutex());
68 if(g_data_manager ==
nullptr) {
72 return g_data_manager;
77 std::lock_guard<std::recursive_mutex> lock(get_mutex());
81 g_data_manager =
this;
86 g_data_manager =
nullptr;
89shared_ptr<tklevel_scheme> tkmanager::get_level_scheme(
const tkn::tkstring &_nuc,
int _zz,
int _aa)
91 if(fmap_of_level_scheme.count(_nuc))
return fmap_of_level_scheme[_nuc];
93 std::lock_guard<std::recursive_mutex> lock(get_mutex());
95 auto [it, inserted] = fmap_of_level_scheme.try_emplace(
96 _nuc, make_shared<tklevel_scheme>(_nuc,_zz,_aa));
100void tkmanager::preload_nuclei()
103 gdatabase->begin(
"isotope.*, element.*",
"isotope INNER JOIN element on isotope.element_id=element.element_id");
105 while(gdatabase->next()) {
107 tkdb_table& iso_table = (*gdatabase)[
"ISOTOPE"];
109 shared_ptr<tknucleus> anucleus = make_shared<tknucleus>();
111 tkstring name = element[
"symbol"].get_value();
112 anucleus->felement_symbol = name;
113 anucleus->felement_name = element[
"name"].get_value();
115 int zz = element[
"charge"].get_value().atoi();
116 int aa = iso_table[
"mass"].get_value().atoi();
118 gdebug << aa << name <<
" loaded" <<
do_endl;
120 name.
prepend(iso_table[
"mass"].get_value());
122 anucleus->set_name(name.data());
125 anucleus->fN = aa - zz;
126 anucleus->fis_known =
true;
131 if(colname==
"element_id")
continue;
134 tkstring colname_unit = colname +
"_unit";
136 tkstring value = col.second.get_value();
138 unit = element.
get_columns().at(colname_unit).get_value();
140 if( colname ==
"atomic_mass" ||
141 colname ==
"atomic_radius_van_der_Waals" ||
142 colname ==
"boiling_point" ||
143 colname ==
"density" ||
144 colname ==
"ionization_energy" ||
145 colname ==
"melting_point"
147 shared_ptr<tkmeasure> data = make_shared<tkmeasure>(value.
atof(),unit);
149 data->set_type(colname);
150 anucleus->add_property(colname,data);
154 shared_ptr<tkmeasure> data = make_shared<tkmeasure>(value.
atof(),unit);
156 data->set_type(colname);
157 if(value.
atof()>0) anucleus->add_property(colname,data);
160 anucleus->add_property_str(colname,value,unit);
165 vector<tkstring> fNames = {
167 "abundance",
"quadrupoleDeformation",
"FY235U",
"FY238U",
"FY239Pu",
"FY241Pu",
"FY252Cf",
"cFY235U",
"cFY238U",
"cFY239Pu",
"cFY241Pu",
"cFY252Cf",
168 "mass_excess",
"radius",
"magnetic_dipole",
"electric_quadrupole"
171 for(
auto &property_name: fNames) {
175 shared_ptr<tkmeasure> data = make_shared<tkmeasure>(data_struct.
value,data_struct.
unit,data_struct.
err);
178 data->set_type(property_name);
180 if(property_name ==
"lifetime") {
182 if(data->get_info_tag().contains(
"STABLE")) {
183 anucleus->fis_stable =
true;
185 anucleus->add_property(property_name,data,
"STABLE",
"");
187 else anucleus->add_property(property_name,data,
tkstring::form(
"%g",data->get_value()));
190 anucleus->add_property(property_name,data);
195 anucleus->add_property_str(
"isotope_year_discovered",iso_table[
"isotope_year_discovered"].get_value(),
"");
197 tkstring decay_modes = iso_table[
"decay_modes"].get_value();
198 anucleus->add_property_str(
"decay_modes",decay_modes,
"");
200 tkstring spin_parity_str = iso_table[
"spin_parity"].get_value();
201 double spin = iso_table[
"spin"].get_value().atof();
202 int parity = iso_table[
"parity"].get_value().atoi();
204 anucleus->fspin_parity.set_spin(spin);
205 anucleus->fspin_parity.set_parity(parity);
206 anucleus->fspin_parity.set_from_str(spin_parity_str);
208 anucleus->add_property_str(
"spin_parity",spin_parity_str,
"");
210 fmap_of_nuclei[name] = anucleus;
211 fmap_of_nuclei_per_z[anucleus->fZ].push_back(anucleus);
212 fmap_of_nuclei_per_a[anucleus->fA].push_back(anucleus);
213 fmap_of_nuclei_per_n[anucleus->fN].push_back(anucleus);
214 fmap_of_nuclei_per_z_and_a[anucleus->fZ][anucleus->fA] = anucleus;
215 fvector_of_nuclei.push_back(anucleus);
217 if(!fmap_of_symbols.count(anucleus->get_z()))
218 fmap_of_symbols[anucleus->get_z()] = anucleus->get_element_symbol();
223 auto neutron_ME = fmap_of_nuclei_per_z_and_a.at(0).at(1)->get(
"mass_excess");
224 auto proton_ME = fmap_of_nuclei_per_z_and_a.at(1).at(1)->get(
"mass_excess");
225 auto alpha_ME = fmap_of_nuclei_per_z_and_a.at(2).at(4)->get(
"mass_excess");
226 tkmeasure electron_mass(0.51099895000,
"MeV",0.00000000015);
227 for(
auto &nuc: fmap_of_nuclei) {
228 auto ME = nuc.second->get(
"mass_excess");
231 auto daughter =
get_nucleus(nuc.second->get_z(),nuc.second->get_a()-1);
232 if(daughter && daughter->get(
"mass_excess")) {
233 shared_ptr<tkmeasure> Sn = make_shared<tkmeasure>(-*ME + *daughter->get(
"mass_excess") + *neutron_ME);
234 Sn->set_type(
"neutronSeparationEnergy");
235 nuc.second->add_property(
"neutronSeparationEnergy",Sn);
238 daughter =
get_nucleus(nuc.second->get_z(),nuc.second->get_a()-2);
239 if(daughter && daughter->get(
"mass_excess")) {
240 shared_ptr<tkmeasure> S2n = make_shared<tkmeasure>(-*ME + *daughter->get(
"mass_excess") + 2.*(*neutron_ME));
241 S2n->set_type(
"twoNeutronSeparationEnergy");
242 nuc.second->add_property(
"twoNeutronSeparationEnergy",S2n);
245 daughter =
get_nucleus(nuc.second->get_z()-1,nuc.second->get_a()-1);
246 if(daughter && daughter->get(
"mass_excess")) {
247 shared_ptr<tkmeasure> Sp = make_shared<tkmeasure>(-*ME + *daughter->get(
"mass_excess") + *proton_ME);
248 Sp->set_type(
"protonSeparationEnergy");
249 nuc.second->add_property(
"protonSeparationEnergy",Sp);
252 daughter =
get_nucleus(nuc.second->get_z()-2,nuc.second->get_a()-2);
253 if(daughter && daughter->get(
"mass_excess")) {
254 shared_ptr<tkmeasure> S2p = make_shared<tkmeasure>(-*ME + *daughter->get(
"mass_excess") + 2.*(*proton_ME));
255 S2p->set_type(
"twoProtonSeparationEnergy");
256 nuc.second->add_property(
"twoProtonSeparationEnergy",S2p);
260 daughter =
get_nucleus(nuc.second->get_z()-2,nuc.second->get_a()-4);
261 if(daughter && daughter->get(
"mass_excess")) {
262 shared_ptr<tkmeasure> Q = make_shared<tkmeasure>( *ME - *daughter->get(
"mass_excess") - *alpha_ME);
263 Q->set_type(
"Qalpha");
264 nuc.second->add_property(
"Qalpha",Q);
267 daughter =
get_nucleus(nuc.second->get_z()+1,nuc.second->get_a());
268 if(daughter && daughter->get(
"mass_excess")) {
269 shared_ptr<tkmeasure> Q = make_shared<tkmeasure>( *ME - *daughter->get(
"mass_excess"));
270 Q->set_type(
"QbetaMinus");
271 nuc.second->add_property(
"QbetaMinus",Q);
274 daughter =
get_nucleus(nuc.second->get_z()+1,nuc.second->get_a()-1);
275 if(daughter && daughter->get(
"mass_excess")) {
276 shared_ptr<tkmeasure> Q = make_shared<tkmeasure>( *ME - *daughter->get(
"mass_excess") - *neutron_ME);
277 Q->set_type(
"QbetaMinusOneNeutronEmission");
278 nuc.second->add_property(
"QbetaMinusOneNeutronEmission",Q);
281 daughter =
get_nucleus(nuc.second->get_z()+1,nuc.second->get_a()-2);
282 if(daughter && daughter->get(
"mass_excess")) {
283 shared_ptr<tkmeasure> Q = make_shared<tkmeasure>( *ME - *daughter->get(
"mass_excess") - 2.*(*neutron_ME) );
284 Q->set_type(
"QbetaMinusTwoNeutronEmission");
285 nuc.second->add_property(
"QbetaMinusTwoNeutronEmission",Q);
288 daughter =
get_nucleus(nuc.second->get_z()+2,nuc.second->get_a()+4);
289 if(daughter && daughter->get(
"mass_excess") && nuc.second->get(
"Qalpha")) {
290 auto Qa = nuc.second->get(
"Qalpha");
291 auto Qa2 = *daughter->get(
"mass_excess") - *ME - *alpha_ME;
292 shared_ptr<tkmeasure> Q = make_shared<tkmeasure>( 0.5*(Qa2 - *Qa ));
293 Q->set_type(
"QdeltaAlpha");
294 nuc.second->add_property(
"QdeltaAlpha",Q);
297 daughter =
get_nucleus(nuc.second->get_z()+2,nuc.second->get_a());
298 if(daughter && daughter->get(
"mass_excess")) {
299 shared_ptr<tkmeasure> Q = make_shared<tkmeasure>( *ME - *daughter->get(
"mass_excess"));
300 Q->set_type(
"QdoubleBetaMinus");
301 nuc.second->add_property(
"QdoubleBetaMinus",Q);
304 daughter =
get_nucleus(nuc.second->get_z()-1,nuc.second->get_a());
305 if(daughter && daughter->get(
"mass_excess")) {
306 shared_ptr<tkmeasure> Q = make_shared<tkmeasure>( *ME - *daughter->get(
"mass_excess"));
307 Q->set_type(
"QelectronCapture");
308 nuc.second->add_property(
"QelectronCapture",Q);
311 daughter =
get_nucleus(nuc.second->get_z()-2,nuc.second->get_a());
312 if(daughter && daughter->get(
"mass_excess")) {
313 shared_ptr<tkmeasure> Q = make_shared<tkmeasure>( *ME - *daughter->get(
"mass_excess"));
314 Q->set_type(
"QdoubleElectronCapture");
315 nuc.second->add_property(
"QdoubleElectronCapture",Q);
318 daughter =
get_nucleus(nuc.second->get_z()-2,nuc.second->get_a()-1);
319 if(daughter && daughter->get(
"mass_excess")) {
320 shared_ptr<tkmeasure> Q = make_shared<tkmeasure>( *ME - *daughter->get(
"mass_excess") - *proton_ME);
321 Q->set_type(
"QelectronCaptureOneProtonEmission");
322 nuc.second->add_property(
"QelectronCaptureOneProtonEmission",Q);
325 daughter =
get_nucleus(nuc.second->get_z()-1,nuc.second->get_a());
326 if(daughter && daughter->get(
"mass_excess")) {
327 auto q_positron = make_shared<tkmeasure>( *ME - *daughter->get(
"mass_excess") - 2.*electron_mass);
328 q_positron->set_type(
"QpositronEmission");
329 nuc.second->add_property(
"QpositronEmission",q_positron);
333 auto q_binding = make_shared<tkmeasure>( 1./nuc.second->get_a()*(nuc.second->get_z()*(*proton_ME) + nuc.second->get_n()*(*neutron_ME) - *ME));
334 q_binding->set_type(
"binding_energy_overA");
335 nuc.second->add_property(
"binding_energy_overA",q_binding);
338 auto nuc_minus =
get_nucleus(nuc.second->get_z(),nuc.second->get_a()-1);
339 auto nuc_plus =
get_nucleus(nuc.second->get_z(),nuc.second->get_a()+1);
340 if(nuc_minus && nuc_plus && nuc_minus->get(
"mass_excess") && nuc_plus->get(
"mass_excess")) {
341 auto q_pairing = make_shared<tkmeasure>( 0.5 * pow(-1,nuc.second->get_n()) * ( -2* (*ME) + *nuc_minus->get(
"mass_excess") + *nuc_plus->get(
"mass_excess")));
342 q_pairing->set_type(
"pairingGap");
343 nuc.second->add_property(
"pairingGap",q_pairing);
350 size_t nnuc=fvector_of_nuclei.size(),inuc=0;
351 for(
const auto &nuc : fvector_of_nuclei){
352 if(_verbose) glog.progress_bar(nnuc,inuc,
"... pre-loading");
353 get_level_scheme(nuc->get_symbol(),nuc->get_z(),nuc->get_a());
373vector<shared_ptr<tknucleus> >
tkmanager::get_nuclei(
const std::function<
bool (shared_ptr<tknucleus>)> &_selection)
375 vector<shared_ptr<tknucleus>> res;
376 std::copy_if(fvector_of_nuclei.begin(), fvector_of_nuclei.end(), std::back_inserter(res),
377 [&_selection](
const auto &nuc) { return _selection(nuc); });
383 for(
auto &nuc_test: fmap_of_nuclei) {
384 if(nuc_test.first.copy().extract_alpha() == symbol) {
385 _z = nuc_test.second->get_z();
394 std::lock_guard<std::recursive_mutex> lock(get_mutex());
396 return fmax_level_id;
401 std::lock_guard<std::recursive_mutex> lock(get_mutex());
403 return fmax_decay_id;
434void tkmanager::load_drip_lines(
const tkstring& json_path)
436 std::ifstream f(json_path);
438 glog <<
error <<
"tkmanager::load_drip_lines: cannot open " << json_path <<
do_endl;
445 }
catch (
const std::exception& e) {
446 glog <<
error <<
"tkmanager::load_drip_lines: JSON parse error in "
447 << json_path <<
" : " << e.what() <<
do_endl;
452 glog <<
error <<
"tkmanager::load_drip_lines: root is not an array in "
459 auto push_if_present = [&](
const json& o,
const char* qname,
const char* sname){
460 if (!o.contains(qname))
return;
461 if (o[qname].is_null())
return;
464 p.
Z = o.value(
"Z", 0);
465 p.
N = o.value(
"N", 0);
466 p.
value_mev = o.value(qname, std::numeric_limits<double>::quiet_NaN());
468 if (o.contains(sname) && !o[sname].is_null()) {
469 p.
sigma_mev = o.value(sname, std::numeric_limits<double>::quiet_NaN());
473 fdrip_lines[p.
type].push_back(std::move(p));
477 for (
const auto& o : j) {
478 if (!o.is_object())
continue;
480 if (!o.contains(
"Z") || !o.contains(
"N"))
continue;
482 push_if_present(o,
"S1n",
"S1n_sigma");
483 push_if_present(o,
"S2n",
"S2n_sigma");
484 push_if_present(o,
"S1p",
"S1p_sigma");
485 push_if_present(o,
"S2p",
"S2p_sigma");
487 }
catch (
const std::exception& e) {
488 glog <<
error <<
"tkmanager::load_drip_lines: error while iterating JSON: "
495 for (
auto& kv : fdrip_lines) {
496 auto& vec = kv.second;
497 std::sort(vec.begin(), vec.end(), [](
const tkn_drip_point& a,
const tkn_drip_point& b){
498 if (a.Z != b.Z) return a.Z < b.Z;
503 glog <<
info <<
"load_drip_lines: loaded " << j.size()
504 <<
" nuclei objects from " << json_path <<
do_endl;
534 if (fdrip_lines.empty()) {
536 load_drip_lines(
tkstring::Form(
"%s/databases/driplines.json", TKN_SYS));
539 auto it = fdrip_lines.find(_type);
540 if (it == fdrip_lines.end()) {
541 glog <<
warning <<
"get_drip_line: unknown type '" << _type <<
"'" <<
do_endl;
545 const auto& vec = it->second;
552 std::unordered_map<int, tkn_drip_point> best_by_Z;
553 for (
const auto& p : vec) {
554 auto itZ = best_by_Z.find(p.
Z);
555 if (itZ == best_by_Z.end()) {
558 const auto& cur = itZ->second;
559 if (std::isnan(cur.value_mev) ||
566 std::vector<tkn_drip_point> reduced;
567 reduced.reserve(best_by_Z.size());
568 std::transform(best_by_Z.begin(), best_by_Z.end(), std::back_inserter(reduced),
569 [](
const auto &kv) { return kv.second; });
571 std::sort(reduced.begin(), reduced.end(), [](
const tkn_drip_point& a,
const tkn_drip_point& b){
572 if (a.Z != b.Z) return a.Z < b.Z;
Representaiton of a sqlite data table.
void read_measure(measure_data_struct &_struct, const tkstring &_col_base_name)
Manages the database loading and provides access to the physics properties.
static tkmanager * the_data_manager()
int get_new_level_id()
define a new unique level id
std::vector< tkn_drip_point > get_drip_line(const std::string &_type, bool reduce_per_Z=true)
Get the drip line for a given type (optionally reduced per Z)
shared_ptr< tknucleus > get_nucleus(const tkstring &_nuc)
return a shared pointer to a nucleus from its name
const vector< shared_ptr< tknucleus > > & get_nuclei()
return a vector containing all the known nuclei
int get_new_decay_id()
define a new unique decay id
bool known_element(tkstring _nuc, int &_z)
is the element symbol is known (ex: "C")
void preload_level_schemes(bool _verbose=false)
preload all the level schemes from the database
std::string with usefull tricks from TString (ROOT) and KVString (KaliVeda) and more....
tkstring extract_alpha()
Returns a tkstring composed only of the alphabetic letters of the original tkstring.
static const char * form(const char *_format,...)
static tkstring Form(const char *_format,...)
bool ends_with(const char *_s, ECaseCompare _cmp=kExact) const
bool contains(const char *_pat, ECaseCompare _cmp=kExact) const
tkstring & prepend(const tkstring &_st)
bool begins_with(const char *_s, ECaseCompare _cmp=kExact) const
double atof() const
Converts a string to double value.
tklog & error(tklog &log)
tklog & do_endl(tklog &log)
tklog & warning(tklog &log)
data structure used to fill a tkmeasure object from the sqlite database
double value_mev
Quantity value (MeV)
std::string type
Quantity type: "S1n", "S2n", "S1p", or "S2p".
double sigma_mev
Quantity uncertainty (MeV)