|
| 1 | +#include <stdexcept> |
| 2 | +#include <cassert> |
| 3 | +#include "monetdb.hh" |
| 4 | +#include <iostream> |
| 5 | + |
| 6 | +#ifndef HAVE_BOOST_REGEX |
| 7 | +#include <regex> |
| 8 | +#else |
| 9 | +#include <boost/regex.hpp> |
| 10 | +using boost::regex; |
| 11 | +using boost::smatch; |
| 12 | +using boost::regex_match; |
| 13 | +#endif |
| 14 | + |
| 15 | +using namespace std; |
| 16 | + |
| 17 | +static regex e_syntax("near \".*\": syntax error"); |
| 18 | +static regex e_user_abort("callback requested query abort"); |
| 19 | + |
| 20 | +extern "C" { |
| 21 | +#include <mapi.h> |
| 22 | +#include <unistd.h> |
| 23 | +} |
| 24 | + |
| 25 | +// connect montetdb |
| 26 | +monetdb_connection::monetdb_connection(std::string &conninfo) |
| 27 | +{ |
| 28 | + dbh = mapi_mapiuri(conninfo.c_str(), "monetdb", "monetdb", "sql"); |
| 29 | + if (mapi_error(dbh)) { |
| 30 | + if (dbh != NULL) { |
| 31 | + mapi_explain(dbh, stderr); |
| 32 | + mapi_destroy(dbh); |
| 33 | + } else { |
| 34 | + fprintf(stderr, "command failed\n"); |
| 35 | + } |
| 36 | + exit(-1); |
| 37 | + } |
| 38 | + mapi_reconnect(dbh); |
| 39 | + if (mapi_error(dbh)) { |
| 40 | + mapi_explain(dbh, stderr); |
| 41 | + mapi_destroy(dbh); |
| 42 | + exit(-1); |
| 43 | + } |
| 44 | +} |
| 45 | + |
| 46 | +// execute queries on MonetDB |
| 47 | +void monetdb_connection::q(const char* query) |
| 48 | +{ |
| 49 | + MapiHdl hdl = mapi_query(dbh, query); |
| 50 | + if (mapi_result_error(hdl) != NULL) |
| 51 | + mapi_explain_result(hdl, stderr); |
| 52 | + mapi_close_handle(hdl); |
| 53 | +} |
| 54 | + |
| 55 | +// disconnect MonetDB |
| 56 | +monetdb_connection::~monetdb_connection() |
| 57 | +{ |
| 58 | + mapi_destroy(dbh); |
| 59 | +} |
| 60 | + |
| 61 | +//load schema from MonetDB |
| 62 | +schema_monetdb::schema_monetdb(std::string &conninfo):monetdb_connection(conninfo) |
| 63 | +{ |
| 64 | + |
| 65 | + |
| 66 | + cerr << "init booltype, inttype, internaltype, arraytype here" << endl; |
| 67 | + booltype = sqltype::get("boolean"); |
| 68 | + inttype = sqltype::get("int"); |
| 69 | + internaltype = sqltype::get("internal"); |
| 70 | + arraytype = sqltype::get("ARRAY"); |
| 71 | + |
| 72 | + cerr << "Loading tables from database: " << conninfo << endl; |
| 73 | +// string qry = "select t.name, s.name, t.system, t.type from sys.tables t, sys.schemas s where t.schema_id=s.id and t.system=false"; |
| 74 | + string qry = "select t.name, s.name, t.system, t.type from sys.tables t, sys.schemas s where t.schema_id=s.id "; |
| 75 | + MapiHdl hdl = mapi_query(dbh,qry.c_str()); |
| 76 | + while (mapi_fetch_row(hdl)) { |
| 77 | + cerr << "."; |
| 78 | + tables.push_back(table(mapi_fetch_field(hdl,0),mapi_fetch_field(hdl,1),strcmp(mapi_fetch_field(hdl,2),"false")==0 ? true : false , atoi(mapi_fetch_field(hdl,3))==0 ? false : true)); |
| 79 | + } |
| 80 | + mapi_close_handle(hdl); |
| 81 | + cerr << " done." << endl; |
| 82 | + |
| 83 | + cerr << "Loading columns and constraints..."; |
| 84 | + for (auto t = tables.begin(); t!=tables.end(); t++) { |
| 85 | + string q("select col.name," |
| 86 | + " col.type " |
| 87 | + " from sys.columns col, sys.tables tab" |
| 88 | + " where tab.name= '"); |
| 89 | + q += t->name; |
| 90 | + q += "' and tab.id = col.table_id"; |
| 91 | + cerr << "."; |
| 92 | + |
| 93 | + hdl = mapi_query(dbh,q.c_str()); |
| 94 | + while (mapi_fetch_row(hdl)) { |
| 95 | + cerr << "."; |
| 96 | + column c(mapi_fetch_field(hdl,0), sqltype::get(mapi_fetch_field(hdl,1))); |
| 97 | + t->columns().push_back(c); |
| 98 | + } |
| 99 | + mapi_close_handle(hdl); |
| 100 | + } |
| 101 | + // TODO: confirm with Martin or Stefan about column |
| 102 | + // constraints in MonetDB |
| 103 | + cerr << " done." << endl; |
| 104 | + |
| 105 | + cerr << "Loading operators..."; |
| 106 | + string opq("select f.func, a.type, b.type, c.type" |
| 107 | + " from sys.functions f, sys.args a, sys.args b, sys.args c" |
| 108 | + " where f.id=a.func_id and f.id=b.func_id and f.id=c.func_id and a.name='arg_1' and b.name='arg_2' and c.number=0"); |
| 109 | + hdl = mapi_query(dbh,opq.c_str()); |
| 110 | + while (mapi_fetch_row(hdl)) { |
| 111 | + cerr << "."; |
| 112 | + op o(mapi_fetch_field(hdl,0),sqltype::get(mapi_fetch_field(hdl,1)),sqltype::get(mapi_fetch_field(hdl,2)),sqltype::get(mapi_fetch_field(hdl,3))); |
| 113 | + register_operator(o); |
| 114 | + } |
| 115 | + mapi_close_handle(hdl); |
| 116 | + cerr << " done." << endl; |
| 117 | + |
| 118 | + |
| 119 | + cerr << "Loading routines..."; |
| 120 | + string routq("select s.name, f.id, a.type, f.name from sys.schemas s, sys.args a, sys.types t, sys.functions f where f.schema_id = s.id and f.id=a.func_id and a.number=0 and a.type = t.sqlname and f.mod<>'aggr'"); |
| 121 | + hdl = mapi_query(dbh,routq.c_str()); |
| 122 | + while (mapi_fetch_row(hdl)) { |
| 123 | + cerr << "."; |
| 124 | + routine proc(mapi_fetch_field(hdl,0),mapi_fetch_field(hdl,1),sqltype::get(mapi_fetch_field(hdl,2)),mapi_fetch_field(hdl,3)); |
| 125 | + register_routine(proc); |
| 126 | + } |
| 127 | + mapi_close_handle(hdl); |
| 128 | + cerr << " done." << endl; |
| 129 | + |
| 130 | + cerr << "Loading routine parameters..."; |
| 131 | + for (auto &proc : routines) { |
| 132 | + string routpq ("select a.type from sys.args a," |
| 133 | + " sys.functions f " |
| 134 | + " where f.id = a.func_id and a.number <> 0 and f.id = '"); |
| 135 | + routpq += proc.specific_name; |
| 136 | + routpq += "'"; |
| 137 | + hdl = mapi_query(dbh,routpq.c_str()); |
| 138 | + while (mapi_fetch_row(hdl)) { |
| 139 | + proc.argtypes.push_back(sqltype::get(mapi_fetch_field(hdl,0))); |
| 140 | + cerr<<"."; |
| 141 | + } |
| 142 | + mapi_close_handle(hdl); |
| 143 | + } |
| 144 | + cerr << " done."<< endl; |
| 145 | + |
| 146 | + |
| 147 | + |
| 148 | + cerr << "Loading aggregates..."; |
| 149 | + string aggq("select s.name, f.id, a.type, f.name from sys.schemas s, sys.args a, sys.types t, sys.functions f where f.schema_id = s.id and f.id=a.func_id and a.number=0 and a.type = t.sqlname and f.mod='aggr'"); |
| 150 | + |
| 151 | + hdl = mapi_query(dbh,aggq.c_str()); |
| 152 | + while (mapi_fetch_row(hdl)) { |
| 153 | + cerr << "."; |
| 154 | + routine proc(mapi_fetch_field(hdl,0),mapi_fetch_field(hdl,1),sqltype::get(mapi_fetch_field(hdl,2)),mapi_fetch_field(hdl,3)); |
| 155 | + register_aggregate(proc); |
| 156 | + } |
| 157 | + mapi_close_handle(hdl); |
| 158 | + cerr << " done." << endl; |
| 159 | + |
| 160 | + cerr << "Loading aggregates parameters..."; |
| 161 | + for (auto &proc: aggregates) { |
| 162 | + string aggpq ("select a.type from sys.args a, sys.functions f " |
| 163 | + "where f.id = a.func_id and a.number <> 0 and f.id = '"); |
| 164 | + aggpq += proc.specific_name; |
| 165 | + aggpq += "'"; |
| 166 | + hdl = mapi_query(dbh,aggpq.c_str()); |
| 167 | + while (mapi_fetch_row(hdl)) { |
| 168 | + proc.argtypes.push_back(sqltype::get(mapi_fetch_field(hdl,0))); |
| 169 | + cerr<<"."; |
| 170 | + } |
| 171 | + mapi_close_handle(hdl); |
| 172 | + } |
| 173 | + cerr << " done."<< endl; |
| 174 | + |
| 175 | + mapi_destroy(dbh); |
| 176 | + generate_indexes(); |
| 177 | + |
| 178 | +// cerr << "print loaded information to check correctness" << endl; |
| 179 | +// cerr << "Loaded tables.... " << endl; |
| 180 | +/* for (auto item : tables) { |
| 181 | + cerr << item.name << "; " << item.schema << "; " << item.is_insertable << "; " << item.is_base_table << endl; |
| 182 | + } |
| 183 | +*/ |
| 184 | +// cerr << "Loaded columns... " << endl; |
| 185 | +/* for (auto tab : tables) { |
| 186 | + for (auto col: tab.columns()) |
| 187 | + cerr << tab.name << "; " << col.name << "; "<<col.type->name << endl; |
| 188 | + } |
| 189 | +*/ |
| 190 | +// cerr << "Loaded aggregates and parameters... " << endl; |
| 191 | +/* for (auto &proc : aggregates) { |
| 192 | + cerr << proc.specific_name << "; " << proc.schema << "; " << proc.name <<"; " << proc.restype->name ; |
| 193 | + for (auto item : proc.argtypes) |
| 194 | + cerr << "; " << item->name; |
| 195 | + cerr << endl; |
| 196 | + } |
| 197 | +*/ |
| 198 | +} |
| 199 | + |
| 200 | +dut_monetdb::dut_monetdb(std::string &conninfo):monetdb_connection(conninfo) |
| 201 | +{ |
| 202 | + //build connection |
| 203 | +} |
| 204 | + |
| 205 | +void dut_monetdb::test(const std::string &stmt) |
| 206 | +{ |
| 207 | + MapiHdl hdl = mapi_query(dbh,"CALL sys.settimeout(1)"); |
| 208 | + mapi_close_handle(hdl); |
| 209 | + cerr << stmt << endl; |
| 210 | + hdl = mapi_query(dbh,stmt.c_str()); |
| 211 | + if (mapi_error(dbh)!=MOK) { |
| 212 | + try { |
| 213 | + if (mapi_error(dbh)==MERROR) { |
| 214 | + mapi_explain_result(hdl, stdout); |
| 215 | + mapi_close_handle(hdl); |
| 216 | + throw dut::syntax("e"); |
| 217 | + } else if (mapi_error(dbh)==MTIMEOUT) { |
| 218 | + mapi_explain_result(hdl, stdout); |
| 219 | + mapi_close_handle(hdl); |
| 220 | + throw dut::timeout("t"); |
| 221 | + } else { |
| 222 | + mapi_explain_result(hdl, stdout); |
| 223 | + mapi_close_handle(hdl); |
| 224 | + throw dut::failure("er"); |
| 225 | + } |
| 226 | + } catch (dut::failure &e) { |
| 227 | + throw dut::failure("et"); |
| 228 | + } |
| 229 | + } |
| 230 | + mapi_close_handle(hdl); |
| 231 | +} |
0 commit comments