ultimatepp/uppsrc/Sql/IntroSch.cpp
cxl 8ebdcbb0d5 uppsrc: NAMESPACE_UPP / END_UPP_NAMESPACE removed
git-svn-id: svn://ultimatepp.org/upp/trunk@10186 f0d560ea-af0d-0410-9eb7-867de7ffcac7
2016-08-26 17:15:30 +00:00

156 lines
3.8 KiB
C++

#include "Sql.h"
namespace Upp {
struct SchTableInfo {
Vector<String> column;
Vector<String> ref_table;
Vector<String> ref_column;
String primary_key;
String prefix;
bool is_table; // It can also be TYPE, which is not in DB
SchTableInfo() { is_table = false; }
};
ArrayMap<String, SchTableInfo>& sSchTableInfo()
{
static ArrayMap<String, SchTableInfo> x;
return x;
}
void SchDbInfoTable(const char *table)
{
sSchTableInfo().GetAdd(table).is_table = true;
}
void SchDbInfoType(const char *table)
{
sSchTableInfo().GetAdd(table).is_table = false;
}
void SchDbInfoColumn(const char *name)
{
SchTableInfo& f = sSchTableInfo().Top();
f.column.Add(ToUpper(f.prefix + name));
f.ref_table.Add();
f.ref_column.Add();
if(IsNull(f.primary_key))
f.primary_key = name;
}
void SchDbInfoVar(void (*fn)(), const char *name)
{
SchTableInfo& f = sSchTableInfo().Top();
String h = f.prefix;
f.prefix << name << '$';
(*fn)();
f.prefix = h;
}
void SchDbInfoReferences(const char *table)
{
sSchTableInfo().Top().ref_table.Top() = table;
}
void SchDbInfoReferences(const char *table, const char *column)
{
SchDbInfoReferences(table);
sSchTableInfo().Top().ref_column.Top() = column;
}
void SchDbInfoPrimaryKey()
{
SchTableInfo& f = sSchTableInfo().Top();
f.primary_key = f.column.Top();
}
void SchDbInfoColumnArray(const char *name, int items)
{
for(int i = 0; i < items; i++)
SchDbInfoColumn(String().Cat() << name << i);
}
const SchTableInfo& GetSchTableInfo(const String& table)
{
static SchTableInfo sSchTableInfoZero;
return sSchTableInfo().Get(~table, sSchTableInfoZero);
}
bool MatchRefName(const String& fk, const String& pk, const String& tab, int phase)
{
return phase || fk == pk || tab + '_' + pk == fk;
}
SqlBool Join(const String& tab1, const String& as1, const String& tab2, const String& as2, int phase)
{
const SchTableInfo& t1 = GetSchTableInfo(tab1);
const SchTableInfo& t2 = GetSchTableInfo(tab2);
for(int i = 0; i < t1.ref_table.GetCount(); i++)
if(t1.ref_table[i] == tab2 && MatchRefName(t1.column[i], t2.primary_key, tab2, phase))
return SqlId(t1.column[i]).Of(SqlId(as1)) == SqlId(t2.primary_key).Of(SqlId(as2));
for(int i = 0; i < t2.ref_table.GetCount(); i++)
if(t2.ref_table[i] == tab1 && MatchRefName(t2.column[i], t1.primary_key, tab1, phase))
return SqlId(t2.column[i]).Of(SqlId(as2)) == SqlId(t1.primary_key).Of(SqlId(as1));
return SqlBool::False();
}
int sChrf(int c)
{
return c < 32 ? ' ' : c;
}
StaticMutex sM;
Vector<String> GetSchTables()
{
Mutex::Lock __(sM);
Vector<String> r;
for(int i = 0; i < sSchTableInfo().GetCount(); i++)
if(sSchTableInfo()[i].is_table)
r.Add(sSchTableInfo().GetKey(i));
return r;
}
Vector<String> GetSchColumns(const String& table)
{
Mutex::Lock __(sM);
return Vector<String>(GetSchTableInfo(table).column, 0);
}
SqlBool FindSchJoin(const String& tables)
{
Mutex::Lock __(sM);
static VectorMap<String, SqlBool> cache;
if(cache.GetCount() > 20000) // Defend against unlikely dynamic Join permutations
cache.Clear();
int q = cache.Find(tables);
if(q >= 0)
return cache[q];
Vector<String> s = Split(Filter(tables, sChrf), ',');
Vector<String> as;
Vector<String> table;
for(int i = 0; i < s.GetCount(); i++) {
Vector<String> ss = Split(s[i], ' ');
if(ss.GetCount()) {
table.Add(ss[0]);
as.Add(ss.Top());
}
}
if(table.GetCount() >= 2) {
String tab1 = table.Top();
String as1 = as.Top();
for(int phase = 0; phase < 2; phase++)
for(int i = 0; i < table.GetCount() - 1; i++) {
SqlBool b = Join(tab1, as1, table[i], as[i], phase);
if(!b.IsFalse()) {
cache.Add(tables, b);
return b;
}
}
}
NEVER_("Schema join not found " + tables);
return SqlBool::False();
}
}