19#include "tkdb_table.h"
20#include "tklevel_scheme.h"
23#include "tkdatabase.h"
50 if(fA<0||fZ<0)
return;
73 vector<shared_ptr<tklevel>> res;
75 std::copy_if(levels.begin(), levels.end(), std::back_inserter(res),
76 [&_selection](
const auto &lvl) { return _selection(lvl); });
96 vector<shared_ptr<tkdecay>> res;
98 std::copy_if(decays.begin(), decays.end(), std::back_inserter(res),
99 [&_selection](
const auto &dec) { return _selection(dec); });
128 glog <<
info <<
" Level scheme of " << fnucleus <<
do_endl;
129 glog <<
info <<
" Options : 'dataset', 'level', 'decay'" <<
do_endl;
132 glog <<
info <<
"Available datasets are : " <<
do_endl;
133 for(
auto &dts : fmap_of_dataset) glog <<
info << dts.second->get_name() <<
" (" << dts.first <<
")" <<
do_endl;
137 fmap_of_dataset[fdatasetid]->print(_data, _option);
140 int _current_data_set = fdatasetid;
141 for(
auto &dataset: fmap_of_dataset) {
142 glog <<
info <<
"dataset : " << dataset.second->get_name() <<
do_endl;
144 dataset.second->print(
"leveldecay");
157 if(fmap_of_dataset_name_id.count(_dataset_name)==0) {
158 glog <<
warning <<
"dataset '" << _dataset_name <<
"' do not exist for " << fnucleus <<
do_endl;
162 fdatasetid = fmap_of_dataset_name_id[_dataset_name];
172 if(fmap_of_dataset.count(_dataset_id)==0) {
173 glog <<
warning <<
"dataset with id '" << _dataset_id <<
"' do not exist for " << fnucleus <<
do_endl;
178 fdatasetid = _dataset_id;
179 fmap_of_dataset[fdatasetid]->load_dataset();
201 const char* sep =
"";
203 else if(_name.
contains(
"-")) {sep =
"-";}
206 if(array.size()!=2) {
207 glog <<
error <<
" <nuclear_level_scheme::get_level_energy()> invalid argument '" << _name <<
"'. Syntax: '[spin][parity][rank]'." <<
do_endl;
212 rank = array.at(1).atoi();
215 for(
const auto &lvlp :
get_levels([](
auto lvl) {
return (lvl->get_spin_parity()!=
nullptr);})) {
216 auto jpistr = lvlp->get_spin_parity()->get_jpi_str();
226 else if(jpistr.equal_to(
tkstring::form(
"%s%s",spin.data(),sep))) {
231 if(idx==rank)
return lvlp;
247 shared_ptr<tklevel> clostest_lev;
248 double best_ediff=1e6;
251 if(_offset != lev->get_offset_bandhead())
continue;
253 if(_offset.
is_empty() && !lev->get_offset_bandhead().is_empty())
continue;
254 if(_offset.
contains(
"dec") && lev->get_decays_down().size()==0)
continue;
256 if(ediff<best_ediff) {
264void tklevel_scheme::init()
266 gdatabase->begin(
"DISTINCT dataset.dataset_id, dataset.dataset_name",
"level INNER JOIN dataset on level.dataset_id=dataset.dataset_id INNER JOIN isotope on level.isotope_id=isotope.isotope_id INNER JOIN element on isotope.element_id=element.element_id",
tkstring::form(
"element.charge=%d AND isotope.mass=%d",fZ,fA),
tkstring::form(
"read_dataset_%s",fnucleus.data()));
268 while(gdatabase->next(
tkstring::form(
"read_dataset_%s",fnucleus.data()))) {
269 tkdb_table& datasets = (*gdatabase)[
"DATASET"];
270 fmap_of_dataset_name_id[datasets[
"dataset_name"].get_value()] = datasets[
"dataset_id"].get_value().atoi();
271 fmap_of_dataset[datasets[
"dataset_id"].get_value().atoi()] = make_shared<tkdataset>(
tkdataset(fnucleus,fZ,fA,datasets[
"dataset_name"].get_value(),datasets[
"dataset_id"].get_value().atoi()));
273 tkstring default_dataset = fnucleus.data();
274 default_dataset +=
" : ADOPTED LEVELS, GAMMAS";
275 gdebug <<
"looking for dataset : " << default_dataset <<
do_endl;
276 bool found = (fmap_of_dataset_name_id.count(default_dataset)>0);
277 gdebug << (found?
"":
"not ") <<
"found " << (found?fmap_of_dataset_name_id[default_dataset.data()]:-1) <<
do_endl;
279 if(found)
select_dataset(fmap_of_dataset_name_id[default_dataset.data()]);
281 for(
const auto &dts : fmap_of_dataset) {
282 glog <<
warning <<
tkstring::form(
"%5s",fnucleus.data()) <<
" -- no 'ADOPTED LEVELS' dataset : '" << dts.second->get_name() <<
"' used instead." <<
do_endl;
287 if(fmap_of_dataset.empty()) {
288 tkstring datasetname =
"EMPTY";
290 fmap_of_dataset_name_id[datasetname] = fdatasetid;
291 fmap_of_dataset[fdatasetid] = make_shared<tkdataset>(tkdataset(fnucleus,fZ,fA,datasetname,fdatasetid));
297void tkdataset::check_yrast(shared_ptr<tklevel> _lvl)
300 if(_lvl->get_spin_parity()->get_spin().is_known()) {
301 int TwoJ = (int)(_lvl->get_spin_parity()->get_spin().get_value()*2+0.5);
302 if(!fyrastmap_exact.count(TwoJ)) {
303 fyrastmap_exact[TwoJ] = _lvl;
304 _lvl->fisYrast_exact =
true;
307 if(!fyrastmap_uncertain.count(TwoJ)) {
308 fyrastmap_uncertain[TwoJ] = _lvl;
309 _lvl->fisYrast_uncertain =
true;
313 else if(!_lvl->get_spin_parity_str().is_empty() && !_lvl->get_spin_parity_str().contains(
",")) {
314 tkstring tmp = _lvl->get_spin_parity_str().copy().remove_all(
"(").remove_all(
")").remove_all(
"+").remove_all(
"-");
317 else TwoJ = tmp.
atoi()*2;
318 if(!fyrastmap_uncertain.count(TwoJ)) {
319 fyrastmap_uncertain[TwoJ] = _lvl;
320 _lvl->fisYrast_uncertain =
true;
326void tkdataset::load_dataset()
331 auto t_0 = std::chrono::high_resolution_clock::now();
332 gdebug <<
"loading levels for dataset '" << fname <<
"' of " << fnucleus <<
do_endl;
335 gdatabase->begin(
"DISTINCT level.*",
"level",
tkstring::form(
"dataset_id=%d",fid));
337 while(gdatabase->next()) {
338 tkdb_table *levels = &(*gdatabase)[
"LEVEL"];
339 tkdb_column *col = &(*levels)[
"level_id"];
342 gmanager->set_max_level_id(
id);
344 tkdb_table::measure_data_struct energy_struc;
347 shared_ptr<tklevel> lvl = make_shared<tklevel>(
id,energy_struc);
348 lvl->fbelongs_to_nucleus = fnucleus;
352 tkstring spin_parity_str =
"";
354 col = &(*levels)[
"level_spin_parity"];
356 col = &(*levels)[
"level_spin"];
358 col = &(*levels)[
"level_parity"];
361 if(!spin_parity_str.
is_empty()) lvl->set_jpi(spin,parity,spin_parity_str);
363 tkdb_table::measure_data_struct lifetime_struc;
365 if(lifetime_struc.
filled) lvl->set_lifetime(lifetime_struc);
367 col = &(*levels)[
"level_comment"];
369 col = &(*levels)[
"level_uncertain"];
372 flevels.push_back(lvl);
376 for(
const auto &lvl: flevels) {
377 fmapoflevels[lvl->get_id()] = lvl;
381 auto t_1 = std::chrono::high_resolution_clock::now();
382 gdebug <<
"loading decays for dataset '" << fname <<
"' of " << fnucleus <<
do_endl;
385 gdatabase->begin(
"DISTINCT decay.*",
"decay INNER JOIN level on decay.level_from_id=level.level_id ",
tkstring::form(
"dataset_id=%d",fid));
387 while(gdatabase->next()) {
389 int level_from = -1.;
392 tkdb_table *decay = &(*gdatabase)[
"DECAY"];
393 tkdb_column *col = &(*decay)[
"decay_id"];
396 gmanager->set_max_decay_id(
id);
398 col = &(*decay)[
"decay_type"];
400 col = &(*decay)[
"level_from_id"];
402 col = &(*decay)[
"level_to_id"];
405 shared_ptr<tkdecay> dec =
nullptr;
407 tkdb_table::measure_data_struct energy_struc;
411 dec = make_shared<tkgammadecay>(
id, level_from, level_to, energy_struc);
412 shared_ptr<tkgammadecay> gamma = dynamic_pointer_cast<tkgammadecay>(dec);
414 tkdb_table::measure_data_struct relative_intensity_struc;
415 decay->
read_measure(relative_intensity_struc,
"decay_relative_intensity");
416 if(relative_intensity_struc.
filled) gamma->set_relative_intensity(relative_intensity_struc);
418 col = &(*decay)[
"decay_multipolarity"];
421 tkdb_table::measure_data_struct mixing_ratio_struc;
422 decay->
read_measure(mixing_ratio_struc,
"decay_mixing_ratio");
423 if(mixing_ratio_struc.
filled) gamma->set_mixing_ratio(mixing_ratio_struc);
425 tkdb_table::measure_data_struct conv_coeff_struc;
426 decay->
read_measure(conv_coeff_struc,
"decay_conv_coeff");
427 if(conv_coeff_struc.
filled) gamma->set_conv_coeff(conv_coeff_struc);
430 tkdb_table::measure_data_struct transprob_struc;
432 if(transprob_struc.
filled) gamma->set_transition_probability(transprob_struc);
433 transprob_struc.
clear();
435 if(transprob_struc.
filled) gamma->set_transition_probability(transprob_struc);
436 transprob_struc.
clear();
438 if(transprob_struc.
filled) gamma->set_transition_probability(transprob_struc);
439 transprob_struc.
clear();
441 if(transprob_struc.
filled) gamma->set_transition_probability(transprob_struc);
445 col = &(*decay)[
"decay_comment"];
447 col = &(*decay)[
"decay_uncertain"];
450 fdecays.push_back(dec);
451 fdecays.back()->set_levels(fmapoflevels.at(level_from),fmapoflevels.at(level_to));
453 for(
auto &dec: fdecays) {
454 fmapoflevels.at(dec->get_level_from_id())->add_decay_down(dec);
455 fmapoflevels.at(dec->get_level_to_id())->add_decay_up(dec);
460 auto t_2 = std::chrono::high_resolution_clock::now();
462 auto dt01 = std::chrono::duration<double, std::milli>(t_1-t_0).count() * 0.001;
463 auto dt12 = std::chrono::duration<double, std::milli>(t_2-t_1).count() * 0.001;
465 gdebug << std::fixed << std::setprecision(4) <<
"time level: "
469 gdebug << std::fixed << std::setprecision(4) <<
"time decays: "
484 glog <<
info <<
"dataset '" << fname <<
"' contains " << flevels.size() <<
" levels" <<
do_endl;
485 glog <<
info <<
"dataset '" << fname <<
"' contains " << fdecays.size() <<
" decays" <<
do_endl;
486 for(
auto &lev : flevels) {
487 if(_option.
contains(
"yrastt") && !lev->is_yrast(
true))
continue;
488 else if(_option.
contains(
"yrast") && !_option.
contains(
"yrastt") && !lev->is_yrast())
continue;
490 for(
auto &dec : lev->get_decays_down()) {
491 cout<<setw(13)<<
""<<
"-> ";dec->print(
"quiet;levto");
497 std::cout << std::setw(15) << left <<
"symbol" << std::setw(15) <<
"levelid"
498 << std::setw(15) << left <<
"energy (keV)"
499 << std::setw(15) << left <<
"energy error"
500 << std::setw(15) << left <<
"spin parity"
501 << std::setw(15) << left <<
"t1/2"
502 << std::setw(15) << left <<
"t1/2 error"
503 << std::setw(15) << left <<
"t1/2 unit" << std::endl;}
504 else glog <<
info <<
"dataset '" << fname <<
"' contains " << flevels.size() <<
" levels:" <<
do_endl;
505 for(
auto &lev : flevels) {
506 if(_option.
contains(
"yrastt") && !lev->is_yrast(
true))
continue;
507 else if(_option.
contains(
"yrast") && !_option.
contains(
"yrastt") && !lev->is_yrast())
continue;
508 if(_option.
contains(
"tab")) std::cout << std::setw(15)<< left << fnucleus;
513 if(_option.
contains(
"tab")) std::cout << std::setw(15) << left <<
"symbol" << std::setw(15) << left <<
"decayid" << std::setw(15) << left <<
"energy (keV)" << std::setw(15) << left <<
"energy error (keV)" << std::endl;
514 else glog <<
info <<
"dataset '" << fname <<
"' contains " << fdecays.size() <<
" decays:" <<
do_endl;
515 for(
auto &dec : fdecays) {
516 if(_option.
contains(
"tab")) std::cout << std::setw(15) << left << fnucleus;
526 double spin_val = -1.;
527 int parity_val = -1.;
529 tkdb_table::measure_data_struct lvl_properties{_ener,_unc,0,0,_unit};
530 shared_ptr<tklevel> lvl = make_shared<tklevel>(gmanager->get_new_level_id(),lvl_properties);
532 spin.
set(_jpistr);parity.
set(_jpistr);
536 lvl->set_jpi(spin_val,parity_val,_jpistr);
539 lvl->fbelongs_to_nucleus = fnucleus;
542 auto it = std::lower_bound(flevels.begin(), flevels.end(), lvl,
543 [](
const std::shared_ptr<tklevel>& a,
const std::shared_ptr<tklevel>& b) {
544 return a->get_energy() < b->get_energy();
546 flevels.insert(it, lvl);
549 fmapoflevels[lvl->get_id()] = lvl;
554shared_ptr<tkgammadecay>
tkdataset::add_gamma_decay(shared_ptr<tklevel> _lvlfrom, shared_ptr<tklevel> _lvlto,
double _ener,
double _unc)
556 if(_ener==0.) _ener = _lvlfrom->get_energy() - _lvlto->get_energy();
558 tkdb_table::measure_data_struct energy_struc{_ener,_unc,0.,0.,
"keV"};
560 shared_ptr<tkgammadecay> decay = make_shared<tkgammadecay>(gmanager->get_new_decay_id(), _lvlfrom->get_id(), _lvlto->get_id(), energy_struc);
561 decay->set_levels(_lvlfrom,_lvlto);
562 fdecays.push_back(decay);
563 _lvlfrom->add_decay_down(decay);
564 _lvlto->add_decay_up(decay);
Stores information on a specific dataset.
shared_ptr< tklevel > add_level(double _ener, double _unc, tkstring _unit, tkstring _jpistr)
manually add a new level to the dataset
shared_ptr< tkgammadecay > add_gamma_decay(shared_ptr< tklevel > _lvlfrom, shared_ptr< tklevel > _lvlto, double _ener=0., double _unc=0.)
add a new decay between two levels to the dataset
friend class tklevel_scheme
tkstring get_value() const
Representaiton of a sqlite data table.
void read_measure(measure_data_struct &_struct, const tkstring &_col_base_name)
tklevel_scheme(const tkstring &_nuc, int _zz, int _aa)
void print(const tkstring &_data="", const tkstring &_option="")
print the level scheme information
const std::vector< shared_ptr< tklevel > > & get_levels()
get the vector containing all the levels
bool select_dataset(const tkstring &_dataset_name)
select a dataset using its name
const shared_ptr< tkdataset > & get_dataset()
returns the current dataset
shared_ptr< tklevel > get_level(const tkstring &_name, bool _exact=true)
get the level corresponding to the given name
const std::vector< shared_ptr< tkdecay > > & get_decays()
get the vector containing all the decays
virtual bool is_parity(tkparity::eparity _parity)
test the parity value
virtual void set(const tkstring &_st)
define the parity from a string
bool is_known() const
to get some information about this data
double get_value()
To get the spin as a double.
void set(int n, int d=1)
define the spin value
std::string with usefull tricks from TString (ROOT) and KVString (KaliVeda) and more....
static const char * form(const char *_format,...)
std::vector< tkstring > tokenize(const tkstring &_delim=" ") const
Create a vector of string separated by at least one delimiter.
tkstring & remove_all(const tkstring &_s1)
int atoi() const
Converts a string to integer value.
bool contains(const char *_pat, 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)
tklog & comment(tklog &log)