arsd/database.d

1531 lines
33 KiB
D

/++
Generic interface for RDBMS access. Use with one of the implementations in [arsd.mysql], [arsd.sqlite], [arsd.postgres], or [arsd.mssql]. I'm sorry the docs are not good, but a little bit goes a long way:
---
auto db = new Sqlite("file.db"); // see the implementations for constructors
// then the interface, for any impl can be as simple as:
foreach(row; db.query("SELECT id, name FROM people")) {
string id = row[0];
string name = row[1];
}
db.query("INSERT INTO people (id, name) VALUES (?, ?)", 5, "Adam");
---
To convert to other types, just use [std.conv.to] since everything comes out of this as simple strings.
+/
module arsd.database;
// I should do a prepared statement as a template string arg
public import std.variant;
import std.string;
public import std.datetime;
/*
Database 2.0 plan, WIP:
// Do I want to do some kind of RAII?
auto database = Database(new MySql("connection info"));
* Prepared statement support
* Queries with separate args whenever we can with consistent interface
* Query returns some typed info when we can.
* ....?
PreparedStatement prepareStatement(string sql);
Might be worth looking at doing the preparations in static ctors
so they are always done once per program...
*/
///
interface Database {
/// Actually implements the query for the database. The query() method
/// below might be easier to use.
ResultSet queryImpl(string sql, Variant[] args...);
/// Escapes data for inclusion into an sql string literal
string escape(string sqlData);
/// query to start a transaction, only here because sqlite is apparently different in syntax...
void startTransaction();
/// Just executes a query. It supports placeholders for parameters
final ResultSet query(T...)(string sql, T t) {
Variant[] args;
foreach(arg; t) {
Variant a;
static if(__traits(compiles, a = arg))
a = arg;
else
a = to!string(t);
args ~= a;
}
return queryImpl(sql, args);
}
/// turns a systime into a value understandable by the target database as a timestamp to be concated into a query. so it should be quoted and escaped etc as necessary
string sysTimeToValue(SysTime);
/// Prepared statement api
/*
PreparedStatement prepareStatement(string sql, int numberOfArguments);
*/
}
import std.stdio;
Ret queryOneColumn(Ret, string file = __FILE__, size_t line = __LINE__, T...)(Database db, string sql, T t) {
auto res = db.query(sql, t);
if(res.empty)
throw new Exception("no row in result", file, line);
auto row = res.front;
return to!Ret(row[0]);
}
struct Query {
ResultSet result;
this(T...)(Database db, string sql, T t) if(T.length!=1 || !is(T[0]==Variant[])) {
result = db.query(sql, t);
}
// Version for dynamic generation of args: (Needs to be a template for coexistence with other constructor.
this(T...)(Database db, string sql, T args) if (T.length==1 && is(T[0] == Variant[])) {
result = db.queryImpl(sql, args);
}
int opApply(T)(T dg) if(is(T == delegate)) {
import std.traits;
foreach(row; result) {
ParameterTypeTuple!dg tuple;
foreach(i, item; tuple) {
tuple[i] = to!(typeof(item))(row[i]);
}
if(auto result = dg(tuple))
return result;
}
return 0;
}
}
struct Row {
package string[] row;
package ResultSet resultSet;
string opIndex(size_t idx, string file = __FILE__, int line = __LINE__) {
if(idx >= row.length)
throw new Exception(text("index ", idx, " is out of bounds on result"), file, line);
return row[idx];
}
string opIndex(string name, string file = __FILE__, int line = __LINE__) {
auto idx = resultSet.getFieldIndex(name);
if(idx >= row.length)
throw new Exception(text("no field ", name, " in result"), file, line);
return row[idx];
}
string toString() {
return to!string(row);
}
string[string] toAA() {
string[string] a;
string[] fn = resultSet.fieldNames();
foreach(i, r; row)
a[fn[i]] = r;
return a;
}
int opApply(int delegate(ref string, ref string) dg) {
foreach(a, b; toAA())
mixin(yield("a, b"));
return 0;
}
string[] toStringArray() {
return row;
}
}
import std.conv;
interface ResultSet {
// name for associative array to result index
int getFieldIndex(string field);
string[] fieldNames();
// this is a range that can offer other ranges to access it
bool empty() @property;
Row front() @property;
void popFront() ;
size_t length() @property;
/* deprecated */ final ResultSet byAssoc() { return this; }
}
class DatabaseException : Exception {
this(string msg, string file = __FILE__, size_t line = __LINE__) {
super(msg, file, line);
}
}
abstract class SqlBuilder { }
class InsertBuilder : SqlBuilder {
private string table;
private string[] fields;
private string[] fieldsSetSql;
private Variant[] values;
///
void setTable(string table) {
this.table = table;
}
/// same as adding the arr as values one by one. assumes DB column name matches AA key.
void addVariablesFromAssociativeArray(in string[string] arr, string[] names...) {
foreach(name; names) {
fields ~= name;
if(name in arr) {
fieldsSetSql ~= "?";
values ~= Variant(arr[name]);
} else {
fieldsSetSql ~= "null";
}
}
}
///
void addVariable(T)(string name, T value) {
fields ~= name;
fieldsSetSql ~= "?";
values ~= Variant(value);
}
/// if you use a placeholder, be sure to [addValueForHandWrittenPlaceholder] immediately
void addFieldWithSql(string name, string sql) {
fields ~= name;
fieldsSetSql ~= sql;
}
/// for addFieldWithSql that includes a placeholder
void addValueForHandWrittenPlaceholder(T)(T value) {
values ~= Variant(value);
}
/// executes the query
auto execute(Database db, string supplementalSql = null) {
return db.queryImpl(this.toSql() ~ supplementalSql, values);
}
string toSql() {
string sql = "INSERT INTO\n";
sql ~= "\t" ~ table ~ " (\n";
foreach(idx, field; fields) {
sql ~= "\t\t" ~ field ~ ((idx != fields.length - 1) ? ",\n" : "\n");
}
sql ~= "\t) VALUES (\n";
foreach(idx, field; fieldsSetSql) {
sql ~= "\t\t" ~ field ~ ((idx != fieldsSetSql.length - 1) ? ",\n" : "\n");
}
sql ~= "\t)\n";
return sql;
}
}
/// WARNING: this is as susceptible to SQL injections as you would be writing it out by hand
class SelectBuilder : SqlBuilder {
string[] fields;
string table;
string[] joins;
string[] wheres;
string[] orderBys;
string[] groupBys;
int limit;
int limitStart;
Variant[string] vars;
void setVariable(T)(string name, T value) {
assert(name.length);
if(name[0] == '?')
name = name[1 .. $];
vars[name] = Variant(value);
}
Database db;
this(Database db = null) {
this.db = db;
}
/*
It would be nice to put variables right here in the builder
?name
will prolly be the syntax, and we'll do a Variant[string] of them.
Anything not translated here will of course be in the ending string too
*/
SelectBuilder cloned() {
auto s = new SelectBuilder(this.db);
s.fields = this.fields.dup;
s.table = this.table;
s.joins = this.joins.dup;
s.wheres = this.wheres.dup;
s.orderBys = this.orderBys.dup;
s.groupBys = this.groupBys.dup;
s.limit = this.limit;
s.limitStart = this.limitStart;
foreach(k, v; this.vars)
s.vars[k] = v;
return s;
}
override string toString() {
string sql = "SELECT ";
// the fields first
{
bool outputted = false;
foreach(field; fields) {
if(outputted)
sql ~= ", ";
else
outputted = true;
sql ~= field; // "`" ~ field ~ "`";
}
}
sql ~= " FROM " ~ table;
if(joins.length) {
foreach(join; joins)
sql ~= " " ~ join;
}
if(wheres.length) {
bool outputted = false;
sql ~= " WHERE ";
foreach(w; wheres) {
if(outputted)
sql ~= " AND ";
else
outputted = true;
sql ~= "(" ~ w ~ ")";
}
}
if(groupBys.length) {
bool outputted = false;
sql ~= " GROUP BY ";
foreach(o; groupBys) {
if(outputted)
sql ~= ", ";
else
outputted = true;
sql ~= o;
}
}
if(orderBys.length) {
bool outputted = false;
sql ~= " ORDER BY ";
foreach(o; orderBys) {
if(outputted)
sql ~= ", ";
else
outputted = true;
sql ~= o;
}
}
if(limit) {
sql ~= " LIMIT ";
if(limitStart)
sql ~= to!string(limitStart) ~ ", ";
sql ~= to!string(limit);
}
if(db is null)
return sql;
return escapedVariants(db, sql, vars);
}
}
// /////////////////////sql//////////////////////////////////
// used in the internal placeholder thing
string toSql(Database db, Variant a) {
auto v = a.peek!(void*);
if(v && (*v is null)) {
return "NULL";
} else if(auto t = a.peek!(SysTime)) {
return db.sysTimeToValue(*t);
} else if(auto t = a.peek!(DateTime)) {
// FIXME: this might be broken cuz of timezones!
return db.sysTimeToValue(cast(SysTime) *t);
} else if(auto t = a.peek!string) {
auto str = *t;
if(str is null)
return "NULL";
else
return '\'' ~ db.escape(str) ~ '\'';
} else {
string str = to!string(a);
return '\'' ~ db.escape(str) ~ '\'';
}
assert(0);
}
// just for convenience; "str".toSql(db);
string toSql(string s, Database db) {
//if(s is null)
//return "NULL";
return '\'' ~ db.escape(s) ~ '\'';
}
string toSql(long s, Database db) {
return to!string(s);
}
string escapedVariants(Database db, in string sql, Variant[string] t) {
if(t.keys.length <= 0 || sql.indexOf("?") == -1) {
return sql;
}
string fixedup;
int currentStart = 0;
// FIXME: let's make ?? render as ? so we have some escaping capability
foreach(i, dchar c; sql) {
if(c == '?') {
fixedup ~= sql[currentStart .. i];
int idxStart = cast(int) i + 1;
int idxLength;
bool isFirst = true;
while(idxStart + idxLength < sql.length) {
char C = sql[idxStart + idxLength];
if((C >= 'a' && C <= 'z') || (C >= 'A' && C <= 'Z') || C == '_' || (!isFirst && C >= '0' && C <= '9'))
idxLength++;
else
break;
isFirst = false;
}
auto idx = sql[idxStart .. idxStart + idxLength];
if(idx in t) {
fixedup ~= toSql(db, t[idx]);
currentStart = idxStart + idxLength;
} else {
// just leave it there, it might be done on another layer
currentStart = cast(int) i;
}
}
}
fixedup ~= sql[currentStart .. $];
return fixedup;
}
/// Note: ?n params are zero based!
string escapedVariants(Database db, in string sql, Variant[] t) {
// FIXME: let's make ?? render as ? so we have some escaping capability
// if nothing to escape or nothing to escape with, don't bother
if(t.length > 0 && sql.indexOf("?") != -1) {
string fixedup;
int currentIndex;
int currentStart = 0;
foreach(i, dchar c; sql) {
if(c == '?') {
fixedup ~= sql[currentStart .. i];
int idx = -1;
currentStart = cast(int) i + 1;
if((i + 1) < sql.length) {
auto n = sql[i + 1];
if(n >= '0' && n <= '9') {
currentStart = cast(int) i + 2;
idx = n - '0';
}
}
if(idx == -1) {
idx = currentIndex;
currentIndex++;
}
if(idx < 0 || idx >= t.length)
throw new Exception("SQL Parameter index is out of bounds: " ~ to!string(idx) ~ " at `"~sql[0 .. i]~"`");
fixedup ~= toSql(db, t[idx]);
}
}
fixedup ~= sql[currentStart .. $];
return fixedup;
/*
string fixedup;
int pos = 0;
void escAndAdd(string str, int q) {
fixedup ~= sql[pos..q] ~ '\'' ~ db.escape(str) ~ '\'';
}
foreach(a; t) {
int q = sql[pos..$].indexOf("?");
if(q == -1)
break;
q += pos;
auto v = a.peek!(void*);
if(v && (*v is null))
fixedup ~= sql[pos..q] ~ "NULL";
else {
string str = to!string(a);
escAndAdd(str, q);
}
pos = q+1;
}
fixedup ~= sql[pos..$];
sql = fixedup;
*/
}
return sql;
}
enum UpdateOrInsertMode {
CheckForMe,
AlwaysUpdate,
AlwaysInsert
}
// BIG FIXME: this should really use prepared statements
int updateOrInsert(Database db, string table, string[string] values, string where, UpdateOrInsertMode mode = UpdateOrInsertMode.CheckForMe, string key = "id") {
string identifierQuote = "";
bool insert = false;
final switch(mode) {
case UpdateOrInsertMode.CheckForMe:
auto res = db.query("SELECT "~key~" FROM "~identifierQuote~db.escape(table)~identifierQuote~" WHERE " ~ where);
insert = res.empty;
break;
case UpdateOrInsertMode.AlwaysInsert:
insert = true;
break;
case UpdateOrInsertMode.AlwaysUpdate:
insert = false;
break;
}
if(insert) {
string insertSql = "INSERT INTO " ~identifierQuote ~ db.escape(table) ~ identifierQuote ~ " ";
bool outputted = false;
string vs, cs;
foreach(column, value; values) {
if(column is null)
continue;
if(outputted) {
vs ~= ", ";
cs ~= ", ";
} else
outputted = true;
//cs ~= "`" ~ db.escape(column) ~ "`";
cs ~= identifierQuote ~ column ~ identifierQuote; // FIXME: possible insecure
if(value is null)
vs ~= "NULL";
else
vs ~= "'" ~ db.escape(value) ~ "'";
}
if(!outputted)
return 0;
insertSql ~= "(" ~ cs ~ ")";
insertSql ~= " VALUES ";
insertSql ~= "(" ~ vs ~ ")";
db.query(insertSql);
return 0; // db.lastInsertId;
} else {
string updateSql = "UPDATE "~identifierQuote~db.escape(table)~identifierQuote~" SET ";
bool outputted = false;
foreach(column, value; values) {
if(column is null)
continue;
if(outputted)
updateSql ~= ", ";
else
outputted = true;
if(value is null)
updateSql ~= identifierQuote ~ db.escape(column) ~ identifierQuote ~ " = NULL";
else
updateSql ~= identifierQuote ~ db.escape(column) ~ identifierQuote ~ " = '" ~ db.escape(value) ~ "'";
}
if(!outputted)
return 0;
updateSql ~= " WHERE " ~ where;
db.query(updateSql);
return 0;
}
}
string fixupSqlForDataObjectUse(string sql, string[string] keyMapping = null) {
string[] tableNames;
string piece = sql;
sizediff_t idx;
while((idx = piece.indexOf("JOIN")) != -1) {
auto start = idx + 5;
auto i = start;
while(piece[i] != ' ' && piece[i] != '\n' && piece[i] != '\t' && piece[i] != ',')
i++;
auto end = i;
tableNames ~= strip(piece[start..end]);
piece = piece[end..$];
}
idx = sql.indexOf("FROM");
if(idx != -1) {
auto start = idx + 5;
auto i = start;
start = i;
while(i < sql.length && !(sql[i] > 'A' && sql[i] <= 'Z')) // if not uppercase, except for A (for AS) to avoid SQL keywords (hack)
i++;
auto from = sql[start..i];
auto pieces = from.split(",");
foreach(p; pieces) {
p = p.strip();
start = 0;
i = 0;
while(i < p.length && p[i] != ' ' && p[i] != '\n' && p[i] != '\t' && p[i] != ',')
i++;
tableNames ~= strip(p[start..i]);
}
string sqlToAdd;
foreach(tbl; tableNames) {
if(tbl.length) {
string keyName = "id";
if(tbl in keyMapping)
keyName = keyMapping[tbl];
sqlToAdd ~= ", " ~ tbl ~ "." ~ keyName ~ " AS " ~ "id_from_" ~ tbl;
}
}
sqlToAdd ~= " ";
sql = sql[0..idx] ~ sqlToAdd ~ sql[idx..$];
}
return sql;
}
/*
This is like a result set
DataObject res = [...];
res.name = "Something";
res.commit; // runs the actual update or insert
res = new DataObject(fields, tables
when doing a select, we need to figure out all the tables and modify the query to include the ids we need
search for FROM and JOIN
the next token is the table name
right before the FROM, add the ids of each table
given:
SELECT name, phone FROM customers LEFT JOIN phones ON customer.id = phones.cust_id
we want:
SELECT name, phone, customers.id AS id_from_customers, phones.id AS id_from_phones FROM customers LEFT JOIN phones ON customer.id[...];
*/
mixin template DataObjectConstructors() {
this(Database db, string[string] res, Tuple!(string, string)[string] mappings) {
super(db, res, mappings);
}
}
string yield(string what) { return `if(auto result = dg(`~what~`)) return result;`; }
import std.typecons;
import std.json; // for json value making
class DataObject {
// lets you just free-form set fields, assuming they all come from the given table
// note it doesn't try to handle joins for new rows. you've gotta do that yourself
this(Database db, string table) {
assert(db !is null);
this.db = db;
this.table = table;
mode = UpdateOrInsertMode.CheckForMe;
}
JSONValue makeJsonValue() {
JSONValue val;
JSONValue[string] valo;
//val.type = JSON_TYPE.OBJECT;
foreach(k, v; fields) {
JSONValue s;
//s.type = JSON_TYPE.STRING;
s.str = v;
valo[k] = s;
}
val = valo;
return val;
}
this(Database db, string[string] res, Tuple!(string, string)[string] mappings) {
this.db = db;
this.mappings = mappings;
this.fields = res;
mode = UpdateOrInsertMode.AlwaysUpdate;
}
string table;
// table, column [alias]
Tuple!(string, string)[string] mappings;
// value [field] [table]
string[string][string] multiTableKeys; // note this is not set internally tight now
// but it can be set manually to do multi table mappings for automatic update
string opDispatch(string field, string file = __FILE__, size_t line = __LINE__)()
if((field.length < 8 || field[0..8] != "id_from_") && field != "popFront")
{
if(field !in fields)
throw new Exception("no such field " ~ field, file, line);
return fields[field];
}
string opDispatch(string field, T)(T t)
if((field.length < 8 || field[0..8] != "id_from_") && field != "popFront")
{
static if(__traits(compiles, t is null)) {
if(t is null)
setImpl(field, null);
else
setImpl(field, to!string(t));
} else
setImpl(field, to!string(t));
return fields[field];
}
private void setImpl(string field, string value) {
if(field in fields) {
if(fields[field] != value)
changed[field] = true;
} else {
changed[field] = true;
}
fields[field] = value;
}
public void setWithoutChange(string field, string value) {
fields[field] = value;
}
int opApply(int delegate(ref string) dg) {
foreach(a; fields)
mixin(yield("a"));
return 0;
}
int opApply(int delegate(ref string, ref string) dg) {
foreach(a, b; fields)
mixin(yield("a, b"));
return 0;
}
string opIndex(string field, string file = __FILE__, size_t line = __LINE__) {
if(field !in fields)
throw new DatabaseException("No such field in data object: " ~ field, file, line);
return fields[field];
}
string opIndexAssign(string value, string field) {
setImpl(field, value);
return value;
}
string* opBinary(string op)(string key) if(op == "in") {
return key in fields;
}
string[string] fields;
bool[string] changed;
void commitChanges() {
commitChanges(cast(string) null, null);
}
void commitChanges(string key, string keyField) {
commitChanges(key is null ? null : [key], keyField is null ? null : [keyField]);
}
void commitChanges(string[] keys, string[] keyFields = null) {
string[string][string] toUpdate;
int updateCount = 0;
foreach(field, c; changed) {
if(c) {
string tbl, col;
if(mappings is null) {
tbl = this.table;
col = field;
} else {
if(field !in mappings)
assert(0, "no such mapping for " ~ field);
auto m = mappings[field];
tbl = m[0];
col = m[1];
}
toUpdate[tbl][col] = fields[field];
updateCount++;
}
}
if(updateCount) {
db.startTransaction();
scope(success) db.query("COMMIT");
scope(failure) db.query("ROLLBACK");
foreach(tbl, values; toUpdate) {
string where, keyFieldToPass;
if(keys is null) {
keys = [null];
}
if(multiTableKeys is null || tbl !in multiTableKeys)
foreach(i, key; keys) {
string keyField;
if(key is null) {
key = "id_from_" ~ tbl;
if(key !in fields)
key = "id";
}
if(i >= keyFields.length || keyFields[i] is null) {
if(key == "id_from_" ~ tbl)
keyField = "id";
else
keyField = key;
} else {
keyField = keyFields[i];
}
if(where.length)
where ~= " AND ";
auto f = key in fields ? fields[key] : null;
if(f is null)
where ~= keyField ~ " = NULL";
else
where ~= keyField ~ " = '"~db.escape(f)~"'" ;
if(keyFieldToPass.length)
keyFieldToPass ~= ", ";
keyFieldToPass ~= keyField;
}
else {
foreach(keyField, v; multiTableKeys[tbl]) {
if(where.length)
where ~= " AND ";
where ~= keyField ~ " = '"~db.escape(v)~"'" ;
if(keyFieldToPass.length)
keyFieldToPass ~= ", ";
keyFieldToPass ~= keyField;
}
}
updateOrInsert(db, tbl, values, where, mode, keyFieldToPass);
}
changed = null;
}
}
void commitDelete() {
if(mode == UpdateOrInsertMode.AlwaysInsert)
throw new Exception("Cannot delete an item not in the database");
assert(table.length); // FIXME, should work with fancy items too
// FIXME: escaping and primary key questions
db.query("DELETE FROM " ~ table ~ " WHERE id = '" ~ db.escape(fields["id"]) ~ "'");
}
string getAlias(string table, string column) {
string ali;
if(mappings is null) {
if(this.table is null) {
mappings[column] = tuple(table, column);
return column;
} else {
assert(table == this.table);
ali = column;
}
} else {
foreach(a, what; mappings)
if(what[0] == table && what[1] == column
&& a.indexOf("id_from_") == -1) {
ali = a;
break;
}
}
return ali;
}
void set(string table, string column, string value) {
string ali = getAlias(table, column);
//assert(ali in fields);
setImpl(ali, value);
}
string select(string table, string column) {
string ali = getAlias(table, column);
//assert(ali in fields);
if(ali in fields)
return fields[ali];
return null;
}
DataObject addNew() {
auto n = new DataObject(db, null);
n.db = this.db;
n.table = this.table;
n.mappings = this.mappings;
foreach(k, v; this.fields)
if(k.indexOf("id_from_") == -1)
n.fields[k] = v;
else
n.fields[k] = null; // don't copy ids
n.mode = UpdateOrInsertMode.AlwaysInsert;
return n;
}
Database db;
UpdateOrInsertMode mode;
}
/**
You can subclass DataObject if you want to
get some compile time checks or better types.
You'll want to disable opDispatch, then forward your
properties to the super opDispatch.
*/
/*mixin*/ string DataObjectField(T, string table, string column, string aliasAs = null)() {
string aliasAs_;
if(aliasAs is null)
aliasAs_ = column;
else
aliasAs_ = aliasAs;
return `
@property void `~aliasAs_~`(`~T.stringof~` setTo) {
super.set("`~table~`", "`~column~`", to!string(setTo));
}
@property `~T.stringof~` `~aliasAs_~` () {
return to!(`~T.stringof~`)(super.select("`~table~`", "`~column~`"));
}
`;
}
mixin template StrictDataObject() {
// disable opdispatch
string opDispatch(string name)(...) if (0) {}
}
string createDataObjectFieldsFromAlias(string table, fieldsToUse)() {
string ret;
fieldsToUse f;
foreach(member; __traits(allMembers, fieldsToUse)) {
ret ~= DataObjectField!(typeof(__traits(getMember, f, member)), table, member);
}
return ret;
}
/**
This creates an editable data object out of a simple struct.
struct MyFields {
int id;
string name;
}
alias SimpleDataObject!("my_table", MyFields) User;
User a = new User(db);
a.id = 30;
a.name = "hello";
a.commitChanges(); // tries an update or insert on the my_table table
Unlike the base DataObject class, this template provides compile time
checking for types and names, based on the struct you pass in:
a.id = "aa"; // compile error
a.notAField; // compile error
*/
class SimpleDataObject(string tableToUse, fieldsToUse) : DataObject {
mixin StrictDataObject!();
mixin(createDataObjectFieldsFromAlias!(tableToUse, fieldsToUse)());
this(Database db) {
super(db, tableToUse);
}
}
/**
Given some SQL, it finds the CREATE TABLE
instruction for the given tableName.
(this is so it can find one entry from
a file with several SQL commands. But it
may break on a complex file, so try to only
feed it simple sql files.)
From that, it pulls out the members to create a
simple struct based on it.
It's not terribly smart, so it will probably
break on complex tables.
Data types handled:
INTEGER, SMALLINT, MEDIUMINT -> D's int
TINYINT -> D's bool
BIGINT -> D's long
TEXT, VARCHAR -> D's string
FLOAT, DOUBLE -> D's double
It also reads DEFAULT values to pass to D, except for NULL.
It ignores any length restrictions.
Bugs:
Skips all constraints
Doesn't handle nullable fields, except with strings
It only handles SQL keywords if they are all caps
This, when combined with SimpleDataObject!(),
can automatically create usable D classes from
SQL input.
*/
struct StructFromCreateTable(string sql, string tableName) {
mixin(getCreateTable(sql, tableName));
}
string getCreateTable(string sql, string tableName) {
skip:
while(readWord(sql) != "CREATE") {}
assert(readWord(sql) == "TABLE");
if(readWord(sql) != tableName)
goto skip;
assert(readWord(sql) == "(");
int state;
int parens;
struct Field {
string name;
string type;
string defaultValue;
}
Field*[] fields;
string word = readWord(sql);
Field* current = new Field(); // well, this is interesting... under new DMD, not using new breaks it in CTFE because it overwrites the one entry!
while(word != ")" || parens) {
if(word == ")") {
parens --;
word = readWord(sql);
continue;
}
if(word == "(") {
parens ++;
word = readWord(sql);
continue;
}
switch(state) {
default: assert(0);
case 0:
if(word[0] >= 'A' && word[0] <= 'Z') {
state = 4;
break; // we want to skip this since it starts with a keyword (we hope)
}
current.name = word;
state = 1;
break;
case 1:
current.type ~= word;
state = 2;
break;
case 2:
if(word == "DEFAULT")
state = 3;
else if (word == ",") {
fields ~= current;
current = new Field();
state = 0; // next
}
break;
case 3:
current.defaultValue = word;
state = 2; // back to skipping
break;
case 4:
if(word == ",")
state = 0;
}
word = readWord(sql);
}
if(current.name !is null)
fields ~= current;
string structCode;
foreach(field; fields) {
structCode ~= "\t";
switch(field.type) {
case "INTEGER":
case "SMALLINT":
case "MEDIUMINT":
structCode ~= "int";
break;
case "BOOLEAN":
case "TINYINT":
structCode ~= "bool";
break;
case "BIGINT":
structCode ~= "long";
break;
case "CHAR":
case "char":
case "VARCHAR":
case "varchar":
case "TEXT":
case "text":
structCode ~= "string";
break;
case "FLOAT":
case "DOUBLE":
structCode ~= "double";
break;
default:
assert(0, "unknown type " ~ field.type ~ " for " ~ field.name);
}
structCode ~= " ";
structCode ~= field.name;
if(field.defaultValue !is null) {
structCode ~= " = " ~ field.defaultValue;
}
structCode ~= ";\n";
}
return structCode;
}
string readWord(ref string src) {
reset:
while(src[0] == ' ' || src[0] == '\t' || src[0] == '\n')
src = src[1..$];
if(src.length >= 2 && src[0] == '-' && src[1] == '-') { // a comment, skip it
while(src[0] != '\n')
src = src[1..$];
goto reset;
}
int start, pos;
if(src[0] == '`') {
src = src[1..$];
while(src[pos] != '`')
pos++;
goto gotit;
}
while(
(src[pos] >= 'A' && src[pos] <= 'Z')
||
(src[pos] >= 'a' && src[pos] <= 'z')
||
(src[pos] >= '0' && src[pos] <= '9')
||
src[pos] == '_'
)
pos++;
gotit:
if(pos == 0)
pos = 1;
string tmp = src[0..pos];
if(src[pos] == '`')
pos++; // skip the ending quote;
src = src[pos..$];
return tmp;
}
/// Combines StructFromCreateTable and SimpleDataObject into a one-stop template.
/// alias DataObjectFromSqlCreateTable(import("file.sql"), "my_table") MyTable;
template DataObjectFromSqlCreateTable(string sql, string tableName) {
alias SimpleDataObject!(tableName, StructFromCreateTable!(sql, tableName)) DataObjectFromSqlCreateTable;
}
/+
class MyDataObject : DataObject {
this() {
super(new Database("localhost", "root", "pass", "social"), null);
}
mixin StrictDataObject!();
mixin(DataObjectField!(int, "users", "id"));
}
void main() {
auto a = new MyDataObject;
a.fields["id"] = "10";
a.id = 34;
a.commitChanges;
}
+/
/*
alias DataObjectFromSqlCreateTable!(import("db.sql"), "users") Test;
void main() {
auto a = new Test(null);
a.cool = "way";
a.value = 100;
}
*/
void typeinfoBugWorkaround() {
assert(0, to!string(typeid(immutable(char[])[immutable(char)[]])));
}
mixin template DatabaseOperations(string table) {
DataObject getAsDb(Database db) {
return objectToDataObject!(typeof(this))(this, db, table);
}
static typeof(this) fromRow(Row row) {
return rowToObject!(typeof(this))(row);
}
static typeof(this) fromId(Database db, long id) {
auto query = new SelectBuilder(db);
query.table = table;
query.fields ~= "*";
query.wheres ~= "id = ?0";
auto res = db.query(query.toString(), id);
if(res.empty)
throw new Exception("no such row");
return fromRow(res.front);
}
}
import std.traits, std.datetime;
enum DbSave;
enum DbNullable;
alias AliasHelper(alias T) = T;
T rowToObject(T)(Row row) {
import arsd.dom, arsd.cgi;
T t;
foreach(memberName; __traits(allMembers, T)) {
alias member = AliasHelper!(__traits(getMember, t, memberName));
foreach(attr; __traits(getAttributes, member)) {
static if(is(attr == DbSave)) {
static if(is(typeof(member) == enum))
__traits(getMember, t, memberName) = cast(typeof(member)) to!int(row[memberName]);
else static if(is(typeof(member) == bool)) {
__traits(getMember, t, memberName) = row[memberName][0] == 't';
} else static if(is(typeof(member) == Html)) {
__traits(getMember, t, memberName).source = row[memberName];
} else static if(is(typeof(member) == DateTime))
__traits(getMember, t, memberName) = cast(DateTime) dTimeToSysTime(to!long(row[memberName]));
else {
if(row[memberName].length)
__traits(getMember, t, memberName) = to!(typeof(member))(row[memberName]);
// otherwise, we'll leave it as .init - most likely null
}
}
}
}
return t;
}
DataObject objectToDataObject(T)(T t, Database db, string table) {
import arsd.dom, arsd.cgi;
DataObject obj = new DataObject(db, table);
foreach(memberName; __traits(allMembers, T)) {
alias member = AliasHelper!(__traits(getMember, t, memberName));
foreach(attr; __traits(getAttributes, member)) {
static if(is(attr == DbSave)) {
static if(is(typeof(member) == enum))
obj.opDispatch!memberName(cast(int) __traits(getMember, t, memberName));
else static if(is(typeof(member) == Html)) {
obj.opDispatch!memberName(__traits(getMember, t, memberName).source);
} else static if(is(typeof(member) == DateTime))
obj.opDispatch!memberName(dateTimeToDTime(__traits(getMember, t, memberName)));
else {
bool done;
foreach(attr2; __traits(getAttributes, member)) {
static if(is(attr2 == DbNullable)) {
if(__traits(getMember, t, memberName) == 0)
done = true;
}
}
if(!done)
obj.opDispatch!memberName(__traits(getMember, t, memberName));
}
}
}
}
return obj;
}
void fillData(T)(string delegate(string, string) setter, T obj, string name) {
fillData( (k, v) { setter(k, v); }, obj, name);
}
void fillData(T)(void delegate(string, string) setter, T obj, string name) {
import arsd.dom, arsd.cgi;
import std.traits;
static if(!isSomeString!T && isArray!T) {
// FIXME: indexing
foreach(o; obj)
fillData(setter, o, name);
} else static if(is(T == DateTime)) {
fillData(setter, obj.toISOExtString(), name);
} else static if(is(T == Html)) {
fillData(setter, obj.source, name);
} else static if(is(T == struct)) {
foreach(idx, memberName; __traits(allMembers, T)) {
alias member = AliasHelper!(__traits(getMember, obj, memberName));
static if(!is(typeof(member) == function))
fillData(setter, __traits(getMember, obj, memberName), name ~ "." ~ memberName);
else static if(is(typeof(member) == function)) {
static if(functionAttributes!member & FunctionAttribute.property) {
fillData(setter, __traits(getMember, obj, memberName)(), name ~ "." ~ memberName);
}
}
}
} else {
auto value = to!string(obj);
setter(name, value);
}
}
struct varchar(size_t max) {
private string payload;
this(string s, string file = __FILE__, size_t line = __LINE__) {
opAssign(s, file, line);
}
typeof(this) opAssign(string s, string file = __FILE__, size_t line = __LINE__) {
if(s.length > max)
throw new Exception(s ~ " :: too long", file, line);
payload = s;
return this;
}
string asString() {
return payload;
}
alias asString this;
}
version (unittest)
{
/// Unittest utility that returns a predefined set of values
package (arsd) final class PredefinedResultSet : ResultSet
{
string[] fields;
Row[] rows;
size_t current;
this(string[] fields, Row[] rows)
{
this.fields = fields;
this.rows = rows;
foreach (ref row; rows)
row.resultSet = this;
}
int getFieldIndex(const string field) const
{
foreach (const idx, const val; fields)
if (val == field)
return cast(int) idx;
assert(false, "No field with name: " ~ field);
}
string[] fieldNames()
{
return fields;
}
@property bool empty() const
{
return current == rows.length;
}
Row front() @property
{
assert(!empty);
return rows[current];
}
void popFront()
{
assert(!empty);
current++;
}
size_t length() @property
{
return rows.length - current;
}
}
}