mirror of https://github.com/adamdruppe/arsd.git
stuff
This commit is contained in:
parent
09fbeea89a
commit
048a3b39d7
98
database.d
98
database.d
|
@ -3,7 +3,23 @@ module arsd.database;
|
||||||
public import std.variant;
|
public import std.variant;
|
||||||
import std.string;
|
import std.string;
|
||||||
|
|
||||||
import core.vararg;
|
/*
|
||||||
|
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 {
|
interface Database {
|
||||||
/// Actually implements the query for the database. The query() method
|
/// Actually implements the query for the database. The query() method
|
||||||
|
@ -16,11 +32,7 @@ interface Database {
|
||||||
/// query to start a transaction, only here because sqlite is apparently different in syntax...
|
/// query to start a transaction, only here because sqlite is apparently different in syntax...
|
||||||
void startTransaction();
|
void startTransaction();
|
||||||
|
|
||||||
// FIXME: this would be better as a template, but can't because it is an interface
|
|
||||||
|
|
||||||
/// Just executes a query. It supports placeholders for parameters
|
/// Just executes a query. It supports placeholders for parameters
|
||||||
/// by using ? in the sql string. NOTE: it only accepts string, int, long, and null types.
|
|
||||||
/// Others will fail runtime asserts.
|
|
||||||
final ResultSet query(T...)(string sql, T t) {
|
final ResultSet query(T...)(string sql, T t) {
|
||||||
Variant[] args;
|
Variant[] args;
|
||||||
foreach(arg; t) {
|
foreach(arg; t) {
|
||||||
|
@ -33,37 +45,12 @@ interface Database {
|
||||||
}
|
}
|
||||||
return queryImpl(sql, args);
|
return queryImpl(sql, args);
|
||||||
}
|
}
|
||||||
version(none)
|
|
||||||
final ResultSet query(string sql, ...) {
|
|
||||||
Variant[] args;
|
|
||||||
foreach(arg; _arguments) {
|
|
||||||
string a;
|
|
||||||
if(arg == typeid(string) || arg == typeid(immutable(string)) || arg == typeid(const(string)))
|
|
||||||
a = va_arg!string(_argptr);
|
|
||||||
else if (arg == typeid(int) || arg == typeid(immutable(int)) || arg == typeid(const(int))) {
|
|
||||||
auto e = va_arg!int(_argptr);
|
|
||||||
a = to!string(e);
|
|
||||||
} else if (arg == typeid(uint) || arg == typeid(immutable(uint)) || arg == typeid(const(uint))) {
|
|
||||||
auto e = va_arg!uint(_argptr);
|
|
||||||
a = to!string(e);
|
|
||||||
} else if (arg == typeid(immutable(char))) {
|
|
||||||
auto e = va_arg!char(_argptr);
|
|
||||||
a = to!string(e);
|
|
||||||
} else if (arg == typeid(long) || arg == typeid(const(long)) || arg == typeid(immutable(long))) {
|
|
||||||
auto e = va_arg!long(_argptr);
|
|
||||||
a = to!string(e);
|
|
||||||
} else if (arg == typeid(ulong) || arg == typeid(const(ulong)) || arg == typeid(immutable(ulong))) {
|
|
||||||
auto e = va_arg!ulong(_argptr);
|
|
||||||
a = to!string(e);
|
|
||||||
} else if (arg == typeid(null)) {
|
|
||||||
a = null;
|
|
||||||
} else assert(0, "invalid type " ~ arg.toString() );
|
|
||||||
|
|
||||||
args ~= Variant(a);
|
/// Prepared statement api
|
||||||
}
|
/*
|
||||||
|
PreparedStatement prepareStatement(string sql, int numberOfArguments);
|
||||||
|
|
||||||
return queryImpl(sql, args);
|
*/
|
||||||
}
|
|
||||||
}
|
}
|
||||||
import std.stdio;
|
import std.stdio;
|
||||||
|
|
||||||
|
@ -710,49 +697,6 @@ class DataObject {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// vararg hack so property assignment works right, even with null
|
|
||||||
version(none)
|
|
||||||
string opDispatch(string field, string file = __FILE__, size_t line = __LINE__)(...)
|
|
||||||
if((field.length < 8 || field[0..8] != "id_from_") && field != "popFront")
|
|
||||||
{
|
|
||||||
if(_arguments.length == 0) {
|
|
||||||
if(field !in fields)
|
|
||||||
throw new Exception("no such field " ~ field, file, line);
|
|
||||||
|
|
||||||
return fields[field];
|
|
||||||
} else if(_arguments.length == 1) {
|
|
||||||
auto arg = _arguments[0];
|
|
||||||
|
|
||||||
string a;
|
|
||||||
if(arg == typeid(string) || arg == typeid(immutable(string)) || arg == typeid(const(immutable(char)[]))) {
|
|
||||||
a = va_arg!(string)(_argptr);
|
|
||||||
} else if (arg == typeid(int) || arg == typeid(immutable(int)) || arg == typeid(const(int))) {
|
|
||||||
auto e = va_arg!(int)(_argptr);
|
|
||||||
a = to!string(e);
|
|
||||||
} else if (arg == typeid(char) || arg == typeid(immutable(char))) {
|
|
||||||
auto e = va_arg!(char)(_argptr);
|
|
||||||
a = to!string(e);
|
|
||||||
} else if (arg == typeid(uint) || arg == typeid(immutable(uint)) || arg == typeid(const(uint))) {
|
|
||||||
auto e = va_arg!uint(_argptr);
|
|
||||||
a = to!string(e);
|
|
||||||
} else if (arg == typeid(long) || arg == typeid(const(long)) || arg == typeid(immutable(long))) {
|
|
||||||
auto e = va_arg!(long)(_argptr);
|
|
||||||
a = to!string(e);
|
|
||||||
} else if (arg == typeid(null)) {
|
|
||||||
a = null;
|
|
||||||
} else assert(0, "invalid type " ~ arg.toString );
|
|
||||||
|
|
||||||
|
|
||||||
auto setTo = a;
|
|
||||||
setImpl(field, setTo);
|
|
||||||
|
|
||||||
return setTo;
|
|
||||||
|
|
||||||
} else assert(0, "too many arguments");
|
|
||||||
|
|
||||||
assert(0); // should never be reached
|
|
||||||
}
|
|
||||||
|
|
||||||
private void setImpl(string field, string value) {
|
private void setImpl(string field, string value) {
|
||||||
if(field in fields) {
|
if(field in fields) {
|
||||||
if(fields[field] != value)
|
if(fields[field] != value)
|
||||||
|
|
2
http2.d
2
http2.d
|
@ -665,7 +665,7 @@ class HttpRequest {
|
||||||
case 2: // reading data
|
case 2: // reading data
|
||||||
auto can = a + bodyReadingState.contentLengthRemaining;
|
auto can = a + bodyReadingState.contentLengthRemaining;
|
||||||
if(can > data.length)
|
if(can > data.length)
|
||||||
can = data.length;
|
can = cast(int) data.length;
|
||||||
|
|
||||||
//if(bodyReadingState.isGzipped || bodyReadingState.isDeflated)
|
//if(bodyReadingState.isGzipped || bodyReadingState.isDeflated)
|
||||||
// responseData.content ~= cast(ubyte[]) uncompress.uncompress(data[a .. can]);
|
// responseData.content ~= cast(ubyte[]) uncompress.uncompress(data[a .. can]);
|
||||||
|
|
39
postgres.d
39
postgres.d
|
@ -7,6 +7,11 @@ import std.string;
|
||||||
import std.exception;
|
import std.exception;
|
||||||
|
|
||||||
// remember to CREATE DATABASE name WITH ENCODING 'utf8'
|
// remember to CREATE DATABASE name WITH ENCODING 'utf8'
|
||||||
|
//
|
||||||
|
// http://www.postgresql.org/docs/8.0/static/libpq-exec.html
|
||||||
|
// ExecParams, PQPrepare, PQExecPrepared
|
||||||
|
//
|
||||||
|
// SQL: `DEALLOCATE name` is how to dealloc a prepared statement.
|
||||||
|
|
||||||
class PostgreSql : Database {
|
class PostgreSql : Database {
|
||||||
// dbname = name is probably the most common connection string
|
// dbname = name is probably the most common connection string
|
||||||
|
@ -23,6 +28,36 @@ class PostgreSql : Database {
|
||||||
PQfinish(conn);
|
PQfinish(conn);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Prepared statement support
|
||||||
|
|
||||||
|
This will be added to the Database interface eventually in some form,
|
||||||
|
but first I need to implement it for all my providers.
|
||||||
|
|
||||||
|
The common function of those 4 will be what I put in the interface.
|
||||||
|
*/
|
||||||
|
|
||||||
|
ResultSet executePreparedStatement(T...)(string name, T args) {
|
||||||
|
char*[args.length] argsStrings;
|
||||||
|
|
||||||
|
foreach(idx, arg; args) {
|
||||||
|
// FIXME: optimize to remove allocations here
|
||||||
|
static if(!is(typeof(arg) == typeof(null)))
|
||||||
|
argsStrings[idx] = toStringz(to!string(arg));
|
||||||
|
// else make it null
|
||||||
|
}
|
||||||
|
|
||||||
|
auto res = PQexecPrepared(conn, toStringz(name), argsStrings.length, argStrings.ptr, 0, null, 0);
|
||||||
|
|
||||||
|
int ress = PQresultStatus(res);
|
||||||
|
if(ress != PGRES_TUPLES_OK
|
||||||
|
&& ress != PGRES_COMMAND_OK)
|
||||||
|
throw new DatabaseException(error());
|
||||||
|
|
||||||
|
return new PostgresResult(res);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
override void startTransaction() {
|
override void startTransaction() {
|
||||||
query("START TRANSACTION");
|
query("START TRANSACTION");
|
||||||
}
|
}
|
||||||
|
@ -183,6 +218,10 @@ extern(C) {
|
||||||
PGresult* PQexec(PGconn*, const char*);
|
PGresult* PQexec(PGconn*, const char*);
|
||||||
void PQclear(PGresult*);
|
void PQclear(PGresult*);
|
||||||
|
|
||||||
|
PGresult* PQprepare(PGconn*, const char* stmtName, const char* query, int nParams, const void* paramTypes);
|
||||||
|
|
||||||
|
PGresult* PQexecPrepared(PGconn*, const char* stmtName, int nParams, const char** paramValues, const int* paramLengths, const int* paramFormats, int resultFormat);
|
||||||
|
|
||||||
int PQresultStatus(PGresult*); // FIXME check return value
|
int PQresultStatus(PGresult*); // FIXME check return value
|
||||||
|
|
||||||
int PQnfields(PGresult*); // number of fields in a result
|
int PQnfields(PGresult*); // number of fields in a result
|
||||||
|
|
41
sslsocket.d
41
sslsocket.d
|
@ -21,11 +21,39 @@ public import std.socket;
|
||||||
// see also:
|
// see also:
|
||||||
// http://msdn.microsoft.com/en-us/library/aa380536%28v=vs.85%29.aspx
|
// http://msdn.microsoft.com/en-us/library/aa380536%28v=vs.85%29.aspx
|
||||||
|
|
||||||
import deimos.openssl.ssl;
|
// import deimos.openssl.ssl;
|
||||||
|
|
||||||
static this() {
|
version=use_openssl;
|
||||||
|
|
||||||
|
version(use_openssl) {
|
||||||
|
alias SslClientSocket = OpenSslSocket;
|
||||||
|
|
||||||
|
extern(C) {
|
||||||
|
int SSL_library_init();
|
||||||
|
void OpenSSL_add_all_ciphers();
|
||||||
|
void OpenSSL_add_all_digests();
|
||||||
|
void SSL_load_error_strings();
|
||||||
|
|
||||||
|
struct SSL {}
|
||||||
|
struct SSL_CTX {}
|
||||||
|
struct SSL_METHOD {}
|
||||||
|
|
||||||
|
SSL_CTX* SSL_CTX_new(const SSL_METHOD* method);
|
||||||
|
SSL* SSL_new(SSL_CTX*);
|
||||||
|
int SSL_set_fd(SSL*, int);
|
||||||
|
int SSL_connect(SSL*);
|
||||||
|
int SSL_write(SSL*, const void*, int);
|
||||||
|
int SSL_read(SSL*, void*, int);
|
||||||
|
void SSL_free(SSL*);
|
||||||
|
void SSL_CTX_free(SSL_CTX*);
|
||||||
|
|
||||||
|
SSL_METHOD* SSLv3_client_method();
|
||||||
|
}
|
||||||
|
|
||||||
|
shared static this() {
|
||||||
SSL_library_init();
|
SSL_library_init();
|
||||||
OpenSSL_add_all_algorithms();
|
OpenSSL_add_all_ciphers();
|
||||||
|
OpenSSL_add_all_digests();
|
||||||
SSL_load_error_strings();
|
SSL_load_error_strings();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -65,8 +93,8 @@ class OpenSslSocket : Socket {
|
||||||
return receive(buf, SocketFlags.NONE);
|
return receive(buf, SocketFlags.NONE);
|
||||||
}
|
}
|
||||||
|
|
||||||
this(AddressFamily af) {
|
this(AddressFamily af, SocketType type = SocketType.STREAM) {
|
||||||
super(af, SocketType.STREAM);
|
super(af, type);
|
||||||
initSsl();
|
initSsl();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -80,10 +108,11 @@ class OpenSslSocket : Socket {
|
||||||
SSL_CTX_free(ctx);
|
SSL_CTX_free(ctx);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
version(ssl_test)
|
version(ssl_test)
|
||||||
void main() {
|
void main() {
|
||||||
auto sock = new OpenSslSocket(AddressFamily.INET);
|
auto sock = new SslClientSocket(AddressFamily.INET);
|
||||||
sock.connect(new InternetAddress("localhost", 443));
|
sock.connect(new InternetAddress("localhost", 443));
|
||||||
sock.send("GET / HTTP/1.0\r\n\r\n");
|
sock.send("GET / HTTP/1.0\r\n\r\n");
|
||||||
import std.stdio;
|
import std.stdio;
|
||||||
|
|
Loading…
Reference in New Issue