mirror of https://github.com/adamdruppe/arsd.git
301 lines
7.3 KiB
D
301 lines
7.3 KiB
D
// NOTE: I haven't even tried to use this for a test yet!
|
|
// It's probably godawful, if it works at all.
|
|
/++
|
|
Implementation of [arsd.database.Database] interface for
|
|
Microsoft SQL Server, via ODBC.
|
|
+/
|
|
module arsd.mssql;
|
|
|
|
version(Windows):
|
|
|
|
pragma(lib, "odbc32");
|
|
|
|
public import arsd.database;
|
|
|
|
import std.string;
|
|
import std.exception;
|
|
|
|
import core.sys.windows.sql;
|
|
import core.sys.windows.sqlext;
|
|
|
|
///
|
|
class MsSql : Database {
|
|
/// auto db = new MsSql("Driver={SQL Server Native Client 10.0};Server=<host>[\\<optional-instance-name>];Database=dbtest;Trusted_Connection=Yes")
|
|
this(string connectionString) {
|
|
SQLAllocHandle(SQL_HANDLE_ENV, cast(void*)SQL_NULL_HANDLE, &env);
|
|
enforce(env !is null);
|
|
scope(failure)
|
|
SQLFreeHandle(SQL_HANDLE_ENV, env);
|
|
SQLSetEnvAttr(env, SQL_ATTR_ODBC_VERSION, cast(void *) SQL_OV_ODBC3, 0);
|
|
SQLAllocHandle(SQL_HANDLE_DBC, env, &conn);
|
|
scope(failure)
|
|
SQLFreeHandle(SQL_HANDLE_DBC, conn);
|
|
enforce(conn !is null);
|
|
|
|
auto ret = SQLDriverConnect(
|
|
conn, null, cast(ubyte*)connectionString.ptr, SQL_NTS,
|
|
null, 0, null,
|
|
SQL_DRIVER_NOPROMPT );
|
|
|
|
if ((ret != SQL_SUCCESS_WITH_INFO) && (ret != SQL_SUCCESS))
|
|
throw new DatabaseException("Unable to connect to ODBC object: " ~ getSQLError(SQL_HANDLE_DBC, conn)); // FIXME: print error
|
|
|
|
//query("SET NAMES 'utf8'"); // D does everything with utf8
|
|
}
|
|
|
|
~this() {
|
|
SQLDisconnect(conn);
|
|
SQLFreeHandle(SQL_HANDLE_DBC, conn);
|
|
SQLFreeHandle(SQL_HANDLE_ENV, env);
|
|
}
|
|
|
|
override void startTransaction() {
|
|
query("START TRANSACTION");
|
|
}
|
|
|
|
// possible fixme, idk if this is right
|
|
override string sysTimeToValue(SysTime s) {
|
|
return "'" ~ escape(s.toISOExtString()) ~ "'";
|
|
}
|
|
|
|
ResultSet queryImpl(string sql, Variant[] args...) {
|
|
sql = escapedVariants(this, sql, args);
|
|
|
|
// this is passed to MsSqlResult to control
|
|
SQLHSTMT statement;
|
|
auto returned = SQLAllocHandle(SQL_HANDLE_STMT, conn, &statement);
|
|
|
|
enforce(returned == SQL_SUCCESS);
|
|
|
|
returned = SQLExecDirect(statement, cast(ubyte*)sql.ptr, cast(SQLINTEGER) sql.length);
|
|
if(returned != SQL_SUCCESS)
|
|
throw new DatabaseException(getSQLError(SQL_HANDLE_STMT, statement));
|
|
|
|
return new MsSqlResult(statement);
|
|
}
|
|
|
|
string escape(string sqlData) { // FIXME
|
|
return ""; //FIX ME
|
|
//return ret.replace("'", "''");
|
|
}
|
|
|
|
string escapeBinaryString(const(ubyte)[] data) { // FIXME
|
|
return "'" ~ escape(cast(string) data) ~ "'";
|
|
}
|
|
|
|
|
|
string error() {
|
|
return null; // FIXME
|
|
}
|
|
|
|
private:
|
|
SQLHENV env;
|
|
SQLHDBC conn;
|
|
}
|
|
|
|
class MsSqlResult : ResultSet {
|
|
// name for associative array to result index
|
|
int getFieldIndex(string field) {
|
|
if(mapping is null)
|
|
makeFieldMapping();
|
|
if (field !in mapping)
|
|
return -1;
|
|
return mapping[field];
|
|
}
|
|
|
|
|
|
string[] fieldNames() {
|
|
if(mapping is null)
|
|
makeFieldMapping();
|
|
return columnNames;
|
|
}
|
|
|
|
// this is a range that can offer other ranges to access it
|
|
bool empty() {
|
|
return isEmpty;
|
|
}
|
|
|
|
Row front() {
|
|
return row;
|
|
}
|
|
|
|
void popFront() {
|
|
if(!isEmpty)
|
|
fetchNext;
|
|
}
|
|
|
|
override size_t length()
|
|
{
|
|
return 1; //FIX ME
|
|
}
|
|
|
|
this(SQLHSTMT statement) {
|
|
this.statement = statement;
|
|
|
|
SQLSMALLINT info;
|
|
SQLNumResultCols(statement, &info);
|
|
numFields = info;
|
|
|
|
fetchNext();
|
|
}
|
|
|
|
~this() {
|
|
SQLFreeHandle(SQL_HANDLE_STMT, statement);
|
|
}
|
|
|
|
private:
|
|
SQLHSTMT statement;
|
|
int[string] mapping;
|
|
string[] columnNames;
|
|
int numFields;
|
|
|
|
bool isEmpty;
|
|
|
|
Row row;
|
|
|
|
void fetchNext() {
|
|
if(isEmpty)
|
|
return;
|
|
|
|
if(SQLFetch(statement) == SQL_SUCCESS) {
|
|
Row r;
|
|
r.resultSet = this;
|
|
DatabaseDatum[] row;
|
|
|
|
for(int i = 0; i < numFields; i++) {
|
|
string a;
|
|
|
|
SQLLEN ptr;
|
|
|
|
more:
|
|
SQLCHAR[1024] buf;
|
|
if(SQLGetData(statement, cast(ushort)(i+1), SQL_CHAR, buf.ptr, 1024, &ptr) != SQL_SUCCESS)
|
|
throw new DatabaseException("get data: " ~ getSQLError(SQL_HANDLE_STMT, statement));
|
|
|
|
assert(ptr != SQL_NO_TOTAL);
|
|
if(ptr == SQL_NULL_DATA)
|
|
a = null;
|
|
else {
|
|
a ~= cast(string) buf[0 .. ptr > 1024 ? 1024 : ptr].idup;
|
|
ptr -= ptr > 1024 ? 1024 : ptr;
|
|
if(ptr)
|
|
goto more;
|
|
}
|
|
row ~= DatabaseDatum(a);
|
|
}
|
|
|
|
r.row = row;
|
|
this.row = r;
|
|
} else {
|
|
isEmpty = true;
|
|
}
|
|
}
|
|
|
|
void makeFieldMapping() {
|
|
for(int i = 0; i < numFields; i++) {
|
|
SQLSMALLINT len;
|
|
SQLCHAR[1024] buf;
|
|
auto ret = SQLDescribeCol(statement,
|
|
cast(ushort)(i+1),
|
|
cast(ubyte*)buf.ptr,
|
|
1024,
|
|
&len,
|
|
null, null, null, null);
|
|
if (ret != SQL_SUCCESS)
|
|
throw new DatabaseException("Field mapping error: " ~ getSQLError(SQL_HANDLE_STMT, statement));
|
|
|
|
string a = cast(string) buf[0 .. len].idup;
|
|
|
|
columnNames ~= a;
|
|
mapping[a] = i;
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
private string getSQLError(short handletype, SQLHANDLE handle)
|
|
{
|
|
char[32] sqlstate;
|
|
char[256] message;
|
|
SQLINTEGER nativeerror=0;
|
|
SQLSMALLINT textlen=0;
|
|
auto ret = SQLGetDiagRec(handletype, handle, 1,
|
|
cast(ubyte*)sqlstate.ptr,
|
|
cast(int*)&nativeerror,
|
|
cast(ubyte*)message.ptr,
|
|
256,
|
|
&textlen);
|
|
|
|
return message[0 .. textlen].idup;
|
|
}
|
|
|
|
/*
|
|
import std.stdio;
|
|
void main() {
|
|
//auto db = new MsSql("Driver={SQL Server};Server=<host>[\\<optional-instance-name>]>;Database=dbtest;Trusted_Connection=Yes");
|
|
auto db = new MsSql("Driver={SQL Server Native Client 10.0};Server=<host>[\\<optional-instance-name>];Database=dbtest;Trusted_Connection=Yes")
|
|
|
|
db.query("INSERT INTO users (id, name) values (30, 'hello mang')");
|
|
|
|
foreach(line; db.query("SELECT * FROM users")) {
|
|
writeln(line[0], line["name"]);
|
|
}
|
|
}
|
|
*/
|
|
|
|
void omg() {
|
|
|
|
enum EMPLOYEE_ID_LEN = 6 ;
|
|
|
|
SQLHENV henv = null;
|
|
SQLHDBC hdbc = null;
|
|
SQLRETURN retcode;
|
|
SQLHSTMT hstmt = null;
|
|
SQLSMALLINT sCustID;
|
|
|
|
SQLCHAR[EMPLOYEE_ID_LEN]szEmployeeID;
|
|
SQL_DATE_STRUCT dsOrderDate;
|
|
SQLINTEGER cbCustID = 0, cbOrderDate = 0, cbEmployeeID = SQL_NTS;
|
|
|
|
retcode = SQLAllocHandle(SQL_HANDLE_ENV, cast(void*) SQL_NULL_HANDLE, &henv);
|
|
retcode = SQLSetEnvAttr(henv, SQL_ATTR_ODBC_VERSION, cast(SQLPOINTER*)SQL_OV_ODBC3, 0);
|
|
|
|
retcode = SQLAllocHandle(SQL_HANDLE_DBC, henv, &hdbc);
|
|
retcode = SQLSetConnectAttr(hdbc, SQL_LOGIN_TIMEOUT, cast(SQLPOINTER)5, 0);
|
|
|
|
retcode = SQLDriverConnect(
|
|
hdbc, null, cast(ubyte*)"DSN=PostgreSQL30Postgres".ptr, SQL_NTS,
|
|
null, 0, null,
|
|
SQL_DRIVER_NOPROMPT );
|
|
|
|
|
|
import std.stdio; writeln(retcode);
|
|
retcode = SQLAllocHandle(SQL_HANDLE_STMT, hdbc, &hstmt);
|
|
|
|
szEmployeeID[0 .. 6] = cast(ubyte[]) "BERGS\0";
|
|
|
|
sCustID = 5;
|
|
dsOrderDate.year = 2006;
|
|
dsOrderDate.month = 3;
|
|
dsOrderDate.day = 17;
|
|
|
|
|
|
/*
|
|
retcode = SQLBindParameter(hstmt, 1, SQL_PARAM_INPUT, SQL_C_CHAR, SQL_CHAR, EMPLOYEE_ID_LEN, 0, szEmployeeID.ptr, 0, &cbEmployeeID);
|
|
import std.stdio; writeln(retcode); writeln(getSQLError(SQL_HANDLE_STMT, hstmt));
|
|
*/
|
|
retcode = SQLBindParameter(hstmt, 1, SQL_PARAM_INPUT, SQL_C_SSHORT, SQL_INTEGER, 0, 0, &sCustID, 0, &cbCustID);
|
|
import std.stdio; writeln(retcode); writeln(getSQLError(SQL_HANDLE_STMT, hstmt));
|
|
/*
|
|
retcode = SQLBindParameter(hstmt, 3, SQL_PARAM_INPUT, SQL_C_TYPE_DATE, SQL_TIMESTAMP, dsOrderDate.sizeof, 0, &dsOrderDate, 0, &cbOrderDate);
|
|
import std.stdio; writeln(retcode); writeln(getSQLError(SQL_HANDLE_STMT, hstmt));
|
|
*/
|
|
|
|
retcode = SQLPrepare(hstmt, cast(SQLCHAR*)"INSERT INTO Orders(CustomerID, EmployeeID, OrderDate) VALUES ('omg', ?, 'now')", SQL_NTS);
|
|
|
|
import std.stdio; writeln("here ", retcode); writeln(getSQLError(SQL_HANDLE_STMT, hstmt));
|
|
|
|
retcode = SQLExecute(hstmt);
|
|
import std.stdio; writeln(retcode); writeln(getSQLError(SQL_HANDLE_STMT, hstmt));
|
|
}
|