Class Reference for E1039 Core & Analysis Software
DbSvc.cc
Go to the documentation of this file.
1 #include <iostream>
2 #include <iomanip>
3 #include <sstream>
4 #include <cstdlib>
5 #include <fstream>
6 #include <wordexp.h> //to expand environmentals
7 #include <TMySQLServer.h>
8 #include <TSQLiteServer.h>
9 #include <TSQLStatement.h>
10 #include <TSQLResult.h>
11 #include <TSQLRow.h>
12 #include <phool/recoConsts.h>
13 #include "DbSvc.h"
14 
15 #define LogInfo(message) std::cout << "DEBUG: " << __FILE__ << " " << __LINE__ << " " << __FUNCTION__ << " ::: " << message << std::endl
16 
17 using namespace std;
18 
19 DbSvc::DbSvc(const SvrId_t svr_id, const UsrId_t usr_id, const std::string my_cnf)
20  : m_svr_id(svr_id)
21  , m_usr_id(usr_id)
22  , m_my_cnf(my_cnf)
23 {
24  SelectServer();
25  SelectUser();
26  ConnectServer();
27 }
28 
29 DbSvc::DbSvc(const SvrId_t svr_id, const std::string dbfile)
30 {
31  m_svr_id = svr_id;
32  if (m_svr_id != LITE) {
33  LogInfo("DbSvc::DbSvc(const SvrId_t, const std::string) is for sqlite db only.");
34  return;
35  }
36 
37  std::string m_dbfile = ExpandEnvironmentals(dbfile);
38  if (!FileExist(m_dbfile)) {
39  LogInfo("SQLITE DB file " << m_dbfile << " does not exist");
40  return;
41  }
42 
43  m_svr = "sqlite://" + m_dbfile;
44  m_con = new TSQLiteServer(m_svr.c_str());
45 }
46 
48 {
49  if (m_con) delete m_con;
50 }
51 
52 void DbSvc::UseSchema(const char* name, const bool do_create, const bool do_drop)
53 {
54  bool found = false;
55  TSQLResult* res = m_con->GetDataBases(name);
56  TSQLRow* row = 0;
57  while ( (row = res->Next()) != 0 ) {
58  string val = row->GetField(0);
59  delete row;
60  if (val == name) found = true;
61  }
62  delete res;
63 
64  int ret = -1; // non-zero
65  if (found) {
66  if (do_drop) {
67  ret = m_con->DropDataBase(name);
68  if (ret == 0) ret = m_con->CreateDataBase(name);
69  } else {
70  ret = 0; // OK just as found.
71  }
72  } else {
73  if (do_create) ret = m_con->CreateDataBase(name);
74  }
75  if (ret == 0) ret = m_con->SelectDataBase(name);
76  if (ret != 0) {
77  cerr << "!!ERROR!! DbSvc::UseSchema: Failed to select '" << name << "'. Abort." << endl;
78  exit(1);
79  }
80 }
81 
82 void DbSvc::DropTable(const char* name)
83 {
84  string query = "drop table if exists ";
85  query += name;
86  if (! m_con->Exec(query.c_str())) {
87  cerr << "!!ERROR!! DbSvc::DropTable(): Failed on " << name << ". Abort." << endl;
88  exit(1);
89  }
90 }
91 
92 bool DbSvc::HasTable(const char* name, const bool exit_on_false)
93 {
94  if (m_con->HasTable(name)) return true;
95  if (exit_on_false) {
96  cerr << "!!ERROR!! DbSvc: The table '" << name << "' does not exist. Abort." << endl;
97  exit(1);
98  }
99  return false;
100 }
101 
102 void DbSvc::CreateTable(const std::string name, const std::vector<std::string> list_var, const std::vector<std::string> list_type, const std::vector<std::string> list_key)
103 {
104  if (HasTable(name)) {
105  cerr << "!!ERROR!! DbSvc::CreateTable(): Table '" << name << "' already exists. To avoid an unintended creation, please check and drop the table in your code. Abort." << endl;
106  exit(1);
107  }
108  if (list_var.size() != list_type.size()) {
109  cerr << "!!ERROR!! DbSvc::CreateTable(): The sizes of the var and type lists don't match. Abort." << endl;
110  exit(1);
111  }
112 
113  ostringstream oss;
114  oss << "create table " << name << "(";
115  for (unsigned int ii = 0; ii < list_var.size(); ii++) {
116  oss << list_var[ii] << " " << list_type[ii] << ", ";
117  }
118  if (list_key.size() == 0) {
119  oss << "primary_id INT not null auto_increment, PRIMARY KEY (primary_id)";
120  } else {
121  oss << "PRIMARY KEY (";
122  for (unsigned int ii = 0; ii < list_key.size(); ii++) {
123  if (ii != 0) oss << ", ";
124  oss << list_key[ii];
125  }
126  oss << ")";
127  }
128  oss << ")";
129  if (! m_con->Exec(oss.str().c_str())) {
130  cerr << "!!ERROR!! DbSvc::CreateTable(): The creation query failed. Abort." << endl;
131  exit(1);
132  }
133 }
134 
135 void DbSvc::CreateTable(const std::string name, const int n_var, const char** list_var, const char** list_type, const int n_key, const char** list_key)
136 {
137  vector<string> vec_var;
138  vector<string> vec_type;
139  for (int ii = 0; ii < n_var; ii++) {
140  vec_var .push_back(list_var [ii]);
141  vec_type.push_back(list_type[ii]);
142  }
143  vector<string> vec_key;
144  for (int ii = 0; ii < n_key; ii++) {
145  vec_key.push_back(list_key[ii]);
146  }
147  CreateTable(name, vec_var, vec_type, vec_key);
148 }
149 
150 void DbSvc::CreateTable(const std::string name, const VarList list)
151 {
152  vector<string> vec_var;
153  vector<string> vec_type;
154  vector<string> vec_key;
155  for (unsigned int ii = 0; ii < list.Size(); ii++) {
156  string name, type;
157  bool is_key;
158  list.Get(ii, name, type, is_key);
159  vec_var .push_back(name);
160  vec_type.push_back(type);
161  if (is_key) vec_key.push_back(name);
162  }
163  CreateTable(name, vec_var, vec_type, vec_key);
164 }
165 
176 TSQLStatement* DbSvc::Process(const char* query)
177 {
178  TSQLStatement* stmt = m_con->Statement(query);
179  if ((! stmt->Process()) && m_svr_id != LITE) { //sqlite returns false if the query returns no data - this should be accepted
180  cerr << "!!ERROR!! DbSvc::Process(): Failed to execute a statement: " << stmt << ".\n"
181  << "Abort." << endl;
182  exit(1);
183  }
184  stmt->StoreResult();
185  return stmt;
186 }
187 
193 void DbSvc::SelectServer()
194 {
195  if (m_svr_id == AutoSvr) {
197  string svr = rc->get_CharFlag("DB_SERVER");
198  if (svr == "DB1" ) m_svr_id = DB1 ;
199  else if (svr == "DB2" ) m_svr_id = DB2 ;
200  else if (svr == "DB3" ) m_svr_id = DB3 ;
201  else if (svr == "DB4" ) m_svr_id = DB4 ;
202  else if (svr == "LOCAL") m_svr_id = LOCAL;
203  else {
204  cerr << "!!ERROR!! DbSvc(): Unsupported DB_SERVER in recoConsts. Abort.\n";
205  exit(1);
206  }
207  }
208  switch (m_svr_id) {
209  case DB1 : m_svr = "e906-db1.fnal.gov"; break;
210  case DB2 : m_svr = "e906-db2.fnal.gov"; break;
211  case DB3 : m_svr = "e906-db3.fnal.gov"; break;
212  case DB4 : m_svr = "e906-db4.fnal.gov"; break;
213  case LOCAL: m_svr = "localhost" ; break;
214  default:
215  cerr << "!!ERROR!! DbSvc(): Unsupported server ID. Abort.\n";
216  exit(1);
217  }
218 }
219 
220 void DbSvc::SelectUser()
221 {
222  if (m_usr_id == AutoUsr) {
224  string usr = rc->get_CharFlag("DB_USER");
225  if (usr == "seaguest" ) m_usr_id = Guest;
226  else if (usr == "production") m_usr_id = Prod ;
227  else {
228  cerr << "!!ERROR!! DbSvc(): Unsupported DB_USER in recoConsts. Abort.\n";
229  exit(1);
230  }
231  }
232 
233  if (m_my_cnf.length() == 0) {
234  if (m_usr_id == Guest) m_my_cnf = "$E1039_RESOURCE/db_conf/guest.cnf";
235  else /* == Prod */ m_my_cnf = "$E1039_RESOURCE/db_conf/prod.cnf";
236  }
237  m_my_cnf = ExpandEnvironmentals(m_my_cnf);
238  //LogInfo("Using "<< m_my_cnf);
239  if (!FileExist(m_my_cnf)) {
240  LogInfo("DB Conf. "<< m_my_cnf << " doesn't exist");
241  }
242 }
243 
244 void DbSvc::ConnectServer()
245 {
246  ostringstream oss;
247  oss << "mysql://" << m_svr << "/?timeout=120&reconnect=1&cnf_file=" << m_my_cnf;
248  string url = oss.str();
249 
250  for (int i_try = 0; i_try < 5; i_try++) { // The connection sometimes fails even with "reconnect=1". Thus let's try it 5 times.
251  m_con = TMySQLServer::Connect(url.c_str(), 0, 0); // User and password must be given in my.cnf, not here.
252  if (m_con && m_con->IsConnected()) {
253  if (i_try > 0) cout << "DbSvc::ConnectServer(): Succeeded at i_try=" << i_try << "." << endl;
254  return; // OK
255  }
256  sleep(10);
257  }
258 
259  cerr << "!!ERROR!! DbSvc::ConnectServer(): Failed. Abort." << endl;
260  exit(1);
261 }
262 
263 bool DbSvc::FileExist(const std::string fileName)
264 {
265  std::ifstream infile(fileName.c_str());
266  return infile.good();
267 }
268 
269 std::string DbSvc::ExpandEnvironmentals( const std::string& input )
270 {
271  // expand any environment variables in the file name
272  wordexp_t exp_result;
273  if(wordexp(input.c_str(), &exp_result, 0) != 0)
274  {
275  std::cout << "ExpandEnvironmentals - ERROR - Your string '" << input << "' cannot be understood!" << endl;
276  return "";
277  }
278  const string output( exp_result.we_wordv[0]);
279  return output;
280 }
281 
282 void DbSvc::VarList::Add(const std::string name, const std::string type, const bool is_key)
283 {
284  m_name .push_back(name);
285  m_type .push_back(type);
286  m_is_key.push_back(is_key);
287 }
288 
289 void DbSvc::VarList::Get(const int idx, std::string& name, std::string& type, bool& is_key) const
290 {
291  name = m_name [idx];
292  type = m_type [idx];
293  is_key = m_is_key[idx];
294 }
#define LogInfo(message)
Definition: DbSvc.cc:15
void Get(const int idx, std::string &name, std::string &type, bool &is_key) const
Definition: DbSvc.cc:289
void Add(const std::string name, const std::string type, const bool is_key=false)
Definition: DbSvc.cc:282
unsigned int Size() const
Definition: DbSvc.h:24
~DbSvc()
Definition: DbSvc.cc:47
DbSvc(const SvrId_t svr_id=AutoSvr, const UsrId_t usr_id=AutoUsr, const std::string my_cnf="")
Definition: DbSvc.cc:19
TSQLStatement * Process(const char *query)
Definition: DbSvc.cc:176
SvrId_t
Definition: DbSvc.h:17
@ LOCAL
Definition: DbSvc.h:17
@ LITE
Definition: DbSvc.h:17
@ DB4
Definition: DbSvc.h:17
@ DB3
Definition: DbSvc.h:17
@ AutoSvr
Definition: DbSvc.h:17
@ DB1
Definition: DbSvc.h:17
@ DB2
Definition: DbSvc.h:17
void DropTable(const char *name)
Definition: DbSvc.cc:82
void UseSchema(const char *name, const bool do_create=false, const bool do_drop=false)
Definition: DbSvc.cc:52
void CreateTable(const std::string name, const std::vector< std::string > list_var, const std::vector< std::string > list_type, const std::vector< std::string > list_key)
Definition: DbSvc.cc:102
UsrId_t
Definition: DbSvc.h:18
@ Guest
Definition: DbSvc.h:18
@ Prod
Definition: DbSvc.h:18
@ AutoUsr
Definition: DbSvc.h:18
bool HasTable(const char *name, const bool exit_on_false=false)
Definition: DbSvc.cc:92
virtual const std::string get_CharFlag(const std::string &flag) const
Definition: PHFlag.cc:13
static recoConsts * instance()
Definition: recoConsts.cc:7