mirror of https://github.com/adamdruppe/arsd.git
dmd 2.061
This commit is contained in:
parent
b6cf1009c1
commit
86d1a3dcb1
21
cgi.d
21
cgi.d
|
@ -53,6 +53,8 @@
|
||||||
+/
|
+/
|
||||||
module arsd.cgi;
|
module arsd.cgi;
|
||||||
|
|
||||||
|
enum long defaultMaxContentLength = 5_000_000;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
||||||
To do a file download offer in the browser:
|
To do a file download offer in the browser:
|
||||||
|
@ -110,7 +112,7 @@ int locationOf(T)(T[] data, string item) {
|
||||||
/// If you are doing a custom cgi class, mixing this in can take care of
|
/// If you are doing a custom cgi class, mixing this in can take care of
|
||||||
/// the required constructors for you
|
/// the required constructors for you
|
||||||
mixin template ForwardCgiConstructors() {
|
mixin template ForwardCgiConstructors() {
|
||||||
this(long maxContentLength = 5_000_000,
|
this(long maxContentLength = defaultMaxContentLength,
|
||||||
string[string] env = null,
|
string[string] env = null,
|
||||||
const(ubyte)[] delegate() readdata = null,
|
const(ubyte)[] delegate() readdata = null,
|
||||||
void delegate(const(ubyte)[]) _rawDataOutput = null,
|
void delegate(const(ubyte)[]) _rawDataOutput = null,
|
||||||
|
@ -208,7 +210,7 @@ class Cgi {
|
||||||
CommandLine }
|
CommandLine }
|
||||||
|
|
||||||
/** Initializes it using a CGI or CGI-like interface */
|
/** Initializes it using a CGI or CGI-like interface */
|
||||||
this(long maxContentLength = 5_000_000,
|
this(long maxContentLength = defaultMaxContentLength,
|
||||||
// use this to override the environment variable listing
|
// use this to override the environment variable listing
|
||||||
in string[string] env = null,
|
in string[string] env = null,
|
||||||
// and this should return a chunk of data. return empty when done
|
// and this should return a chunk of data. return empty when done
|
||||||
|
@ -1613,7 +1615,7 @@ Cgi dummyCgi(Cgi.RequestMethod method = Cgi.RequestMethod.GET, string url = null
|
||||||
env["CONTENT_LENGTH"] = to!string(data.length);
|
env["CONTENT_LENGTH"] = to!string(data.length);
|
||||||
|
|
||||||
auto cgi = new Cgi(
|
auto cgi = new Cgi(
|
||||||
5_000_000,
|
0,
|
||||||
env,
|
env,
|
||||||
{ return data; },
|
{ return data; },
|
||||||
outputSink,
|
outputSink,
|
||||||
|
@ -1900,8 +1902,8 @@ string toHex(int num) {
|
||||||
// the generic mixins
|
// the generic mixins
|
||||||
|
|
||||||
/// Use this instead of writing your own main
|
/// Use this instead of writing your own main
|
||||||
mixin template GenericMain(alias fun, T...) {
|
mixin template GenericMain(alias fun, long maxContentLength = defaultMaxContentLength) {
|
||||||
mixin CustomCgiMain!(Cgi, fun, T);
|
mixin CustomCgiMain!(Cgi, fun, maxContentLength);
|
||||||
}
|
}
|
||||||
|
|
||||||
private string simpleHtmlEncode(string s) {
|
private string simpleHtmlEncode(string s) {
|
||||||
|
@ -1951,7 +1953,7 @@ bool handleException(Cgi cgi, Throwable t) {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// If you want to use a subclass of Cgi with generic main, use this mixin.
|
/// If you want to use a subclass of Cgi with generic main, use this mixin.
|
||||||
mixin template CustomCgiMain(CustomCgi, alias fun, T...) if(is(CustomCgi : Cgi)) {
|
mixin template CustomCgiMain(CustomCgi, alias fun, long maxContentLength = defaultMaxContentLength) if(is(CustomCgi : Cgi)) {
|
||||||
// kinda hacky - the T... is passed to Cgi's constructor in standard cgi mode, and ignored elsewhere
|
// kinda hacky - the T... is passed to Cgi's constructor in standard cgi mode, and ignored elsewhere
|
||||||
|
|
||||||
void main(string[] args) {
|
void main(string[] args) {
|
||||||
|
@ -2092,7 +2094,7 @@ mixin template CustomCgiMain(CustomCgi, alias fun, T...) if(is(CustomCgi : Cgi))
|
||||||
|
|
||||||
Cgi cgi;
|
Cgi cgi;
|
||||||
try {
|
try {
|
||||||
cgi = new CustomCgi(5_000_000, headers, &getScgiChunk, &writeScgi, &flushScgi);
|
cgi = new CustomCgi(maxContentLength, headers, &getScgiChunk, &writeScgi, &flushScgi);
|
||||||
} catch(Throwable t) {
|
} catch(Throwable t) {
|
||||||
sendAll(connection, plainHttpError(true, "400 Bad Request", t));
|
sendAll(connection, plainHttpError(true, "400 Bad Request", t));
|
||||||
connection.close();
|
connection.close();
|
||||||
|
@ -2151,7 +2153,7 @@ mixin template CustomCgiMain(CustomCgi, alias fun, T...) if(is(CustomCgi : Cgi))
|
||||||
|
|
||||||
Cgi cgi;
|
Cgi cgi;
|
||||||
try {
|
try {
|
||||||
cgi = new CustomCgi(5_000_000, fcgienv, &getFcgiChunk, &writeFcgi, &flushFcgi);
|
cgi = new CustomCgi(maxContentLength, fcgienv, &getFcgiChunk, &writeFcgi, &flushFcgi);
|
||||||
} catch(Throwable t) {
|
} catch(Throwable t) {
|
||||||
FCGX_PutStr(cast(ubyte*) t.msg.ptr, t.msg.length, error);
|
FCGX_PutStr(cast(ubyte*) t.msg.ptr, t.msg.length, error);
|
||||||
writeFcgi(cast(const(ubyte)[]) plainHttpError(true, "400 Bad Request", t));
|
writeFcgi(cast(const(ubyte)[]) plainHttpError(true, "400 Bad Request", t));
|
||||||
|
@ -2174,7 +2176,7 @@ mixin template CustomCgiMain(CustomCgi, alias fun, T...) if(is(CustomCgi : Cgi))
|
||||||
// standard CGI is the default version
|
// standard CGI is the default version
|
||||||
Cgi cgi;
|
Cgi cgi;
|
||||||
try {
|
try {
|
||||||
cgi = new CustomCgi(T);
|
cgi = new CustomCgi(maxContentLength);
|
||||||
} catch(Throwable t) {
|
} catch(Throwable t) {
|
||||||
stderr.writeln(t.msg);
|
stderr.writeln(t.msg);
|
||||||
// the real http server will probably handle this;
|
// the real http server will probably handle this;
|
||||||
|
@ -2257,6 +2259,7 @@ void hackAroundLinkerError() {
|
||||||
writeln(typeid(const(immutable(char)[][])[immutable(char)[]]));
|
writeln(typeid(const(immutable(char)[][])[immutable(char)[]]));
|
||||||
writeln(typeid(immutable(char)[][][immutable(char)[]]));
|
writeln(typeid(immutable(char)[][][immutable(char)[]]));
|
||||||
writeln(typeid(Cgi.UploadedFile[immutable(char)[]]));
|
writeln(typeid(Cgi.UploadedFile[immutable(char)[]]));
|
||||||
|
writeln(typeid(Cgi.UploadedFile[][immutable(char)[]]));
|
||||||
writeln(typeid(immutable(Cgi.UploadedFile)[immutable(char)[]]));
|
writeln(typeid(immutable(Cgi.UploadedFile)[immutable(char)[]]));
|
||||||
writeln(typeid(immutable(Cgi.UploadedFile[])[immutable(char)[]]));
|
writeln(typeid(immutable(Cgi.UploadedFile[])[immutable(char)[]]));
|
||||||
writeln(typeid(immutable(char[])[immutable(char)[]]));
|
writeln(typeid(immutable(char[])[immutable(char)[]]));
|
||||||
|
|
2
color.d
2
color.d
|
@ -39,7 +39,7 @@ struct Color {
|
||||||
if(a == 255)
|
if(a == 255)
|
||||||
return format("#%02x%02x%02x", r, g, b);
|
return format("#%02x%02x%02x", r, g, b);
|
||||||
else
|
else
|
||||||
return format("rgba(%d, %d, %d, %f)", r, g, b, cast(real)a / 255.0);
|
return format("rgba(%d, %d, %d, %s)", r, g, b, cast(real)a / 255.0);
|
||||||
}
|
}
|
||||||
|
|
||||||
string toString() {
|
string toString() {
|
||||||
|
|
1
curl.d
1
curl.d
|
@ -65,6 +65,7 @@ struct CurlOptions {
|
||||||
}
|
}
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
//import std.digest.md;
|
||||||
import std.md5;
|
import std.md5;
|
||||||
import std.file;
|
import std.file;
|
||||||
/// this automatically caches to a local file for the given time. it ignores the expires header in favor of your time to keep.
|
/// this automatically caches to a local file for the given time. it ignores the expires header in favor of your time to keep.
|
||||||
|
|
118
database.d
118
database.d
|
@ -167,8 +167,28 @@ class SelectBuilder : SqlBuilder {
|
||||||
int limit;
|
int limit;
|
||||||
int limitStart;
|
int limitStart;
|
||||||
|
|
||||||
|
Variant[string] vars;
|
||||||
|
void setVariable(T)(string name, T value) {
|
||||||
|
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() {
|
SelectBuilder cloned() {
|
||||||
auto s = new SelectBuilder();
|
auto s = new SelectBuilder(this.db);
|
||||||
s.fields = this.fields.dup;
|
s.fields = this.fields.dup;
|
||||||
s.table = this.table;
|
s.table = this.table;
|
||||||
s.joins = this.joins.dup;
|
s.joins = this.joins.dup;
|
||||||
|
@ -178,6 +198,9 @@ class SelectBuilder : SqlBuilder {
|
||||||
s.limit = this.limit;
|
s.limit = this.limit;
|
||||||
s.limitStart = this.limitStart;
|
s.limitStart = this.limitStart;
|
||||||
|
|
||||||
|
foreach(k, v; this.vars)
|
||||||
|
s.vars[k] = v;
|
||||||
|
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -247,31 +270,88 @@ class SelectBuilder : SqlBuilder {
|
||||||
sql ~= to!string(limit);
|
sql ~= to!string(limit);
|
||||||
}
|
}
|
||||||
|
|
||||||
return sql;
|
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 {
|
||||||
|
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(int i, dchar c; sql) {
|
||||||
|
if(c == '?') {
|
||||||
|
fixedup ~= sql[currentStart .. i];
|
||||||
|
|
||||||
|
int idxStart = 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 = i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fixedup ~= sql[currentStart .. $];
|
||||||
|
|
||||||
|
return fixedup;
|
||||||
|
}
|
||||||
|
|
||||||
/// Note: ?n params are zero based!
|
/// Note: ?n params are zero based!
|
||||||
string escapedVariants(Database db, in string sql, Variant[] t) {
|
string escapedVariants(Database db, in string sql, Variant[] t) {
|
||||||
|
// FIXME: let's make ?? render as ? so we have some escaping capability
|
||||||
string toSql(Variant a) {
|
|
||||||
auto v = a.peek!(void*);
|
|
||||||
if(v && (*v is null))
|
|
||||||
return "NULL";
|
|
||||||
else {
|
|
||||||
string str = to!string(a);
|
|
||||||
return '\'' ~ db.escape(str) ~ '\'';
|
|
||||||
}
|
|
||||||
|
|
||||||
assert(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// if nothing to escape or nothing to escape with, don't bother
|
// if nothing to escape or nothing to escape with, don't bother
|
||||||
if(t.length > 0 && sql.indexOf("?") != -1) {
|
if(t.length > 0 && sql.indexOf("?") != -1) {
|
||||||
string fixedup;
|
string fixedup;
|
||||||
|
@ -298,7 +378,7 @@ string escapedVariants(Database db, in string sql, Variant[] t) {
|
||||||
if(idx < 0 || idx >= t.length)
|
if(idx < 0 || idx >= t.length)
|
||||||
throw new Exception("SQL Parameter index is out of bounds: " ~ to!string(idx) ~ " at `"~sql[0 .. i]~"`");
|
throw new Exception("SQL Parameter index is out of bounds: " ~ to!string(idx) ~ " at `"~sql[0 .. i]~"`");
|
||||||
|
|
||||||
fixedup ~= toSql(t[idx]);
|
fixedup ~= toSql(db, t[idx]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
21
dom.d
21
dom.d
|
@ -1945,10 +1945,10 @@ class Element {
|
||||||
return new ElementStream(this);
|
return new ElementStream(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// I moved these from Form because they are generally useful.
|
// I moved these from Form because they are generally useful.
|
||||||
// Ideally, I'd put them in arsd.html and use UFCS, but that doesn't work with the opDispatch here.
|
// Ideally, I'd put them in arsd.html and use UFCS, but that doesn't work with the opDispatch here.
|
||||||
/// Tags: HTML, HTML5
|
/// Tags: HTML, HTML5
|
||||||
|
// FIXME: add overloads for other label types...
|
||||||
Element addField(string label, string name, string type = "text", FormFieldOptions fieldOptions = FormFieldOptions.none) {
|
Element addField(string label, string name, string type = "text", FormFieldOptions fieldOptions = FormFieldOptions.none) {
|
||||||
auto fs = this;
|
auto fs = this;
|
||||||
auto i = fs.addChild("label");
|
auto i = fs.addChild("label");
|
||||||
|
@ -1968,6 +1968,25 @@ class Element {
|
||||||
return i;
|
return i;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Element addField(Element label, string name, string type = "text", FormFieldOptions fieldOptions = FormFieldOptions.none) {
|
||||||
|
auto fs = this;
|
||||||
|
auto i = fs.addChild("label");
|
||||||
|
i.addChild(label);
|
||||||
|
Element input;
|
||||||
|
if(type == "textarea")
|
||||||
|
input = i.addChild("textarea").
|
||||||
|
setAttribute("name", name).
|
||||||
|
setAttribute("rows", "6");
|
||||||
|
else
|
||||||
|
input = i.addChild("input").
|
||||||
|
setAttribute("name", name).
|
||||||
|
setAttribute("type", type);
|
||||||
|
|
||||||
|
// these are html 5 attributes; you'll have to implement fallbacks elsewhere. In Javascript or maybe I'll add a magic thing to html.d later.
|
||||||
|
fieldOptions.applyToElement(input);
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
|
||||||
Element addField(string label, string name, FormFieldOptions fieldOptions) {
|
Element addField(string label, string name, FormFieldOptions fieldOptions) {
|
||||||
return addField(label, name, "text", fieldOptions);
|
return addField(label, name, "text", fieldOptions);
|
||||||
}
|
}
|
||||||
|
|
8
email.d
8
email.d
|
@ -60,7 +60,7 @@ class EmailMessage {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
string toString() {
|
override string toString() {
|
||||||
string boundary = "0016e64be86203dd36047610926a"; // FIXME
|
string boundary = "0016e64be86203dd36047610926a"; // FIXME
|
||||||
|
|
||||||
assert(!isHtml || (isHtml && isMime));
|
assert(!isHtml || (isHtml && isMime));
|
||||||
|
@ -137,6 +137,8 @@ class EmailMessage {
|
||||||
|
|
||||||
void send(RelayInfo mailServer = RelayInfo("smtp://localhost")) {
|
void send(RelayInfo mailServer = RelayInfo("smtp://localhost")) {
|
||||||
auto smtp = new SMTP(mailServer.server);
|
auto smtp = new SMTP(mailServer.server);
|
||||||
|
if(mailServer.username.length)
|
||||||
|
smtp.setAuthentication(mailServer.username, mailServer.password);
|
||||||
const(char)[][] allRecipients = cast(const(char)[][]) (to ~ cc ~ bcc); // WTF cast
|
const(char)[][] allRecipients = cast(const(char)[][]) (to ~ cc ~ bcc); // WTF cast
|
||||||
smtp.mailTo(allRecipients);
|
smtp.mailTo(allRecipients);
|
||||||
smtp.mailFrom = from;
|
smtp.mailFrom = from;
|
||||||
|
@ -145,11 +147,11 @@ class EmailMessage {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void email(string to, string subject, string message, string from) {
|
void email(string to, string subject, string message, string from, RelayInfo mailServer = RelayInfo("smtp://localhost")) {
|
||||||
auto msg = new EmailMessage();
|
auto msg = new EmailMessage();
|
||||||
msg.from = from;
|
msg.from = from;
|
||||||
msg.to = [to];
|
msg.to = [to];
|
||||||
msg.subject = subject;
|
msg.subject = subject;
|
||||||
msg.textBody = message;
|
msg.textBody = message;
|
||||||
msg.send();
|
msg.send(mailServer);
|
||||||
}
|
}
|
||||||
|
|
8
html.d
8
html.d
|
@ -317,6 +317,9 @@ Form makePostLink(string href, Element submitButtonContents, string[string] para
|
||||||
return makePostLink_impl(href, params, submit);
|
return makePostLink_impl(href, params, submit);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
import arsd.cgi;
|
||||||
|
import std.range;
|
||||||
|
|
||||||
Form makePostLink_impl(string href, string[string] params, Element submitButton) {
|
Form makePostLink_impl(string href, string[string] params, Element submitButton) {
|
||||||
auto form = require!Form(Element.make("form"));
|
auto form = require!Form(Element.make("form"));
|
||||||
form.method = "POST";
|
form.method = "POST";
|
||||||
|
@ -791,6 +794,8 @@ mixin template opDispatches(R) {
|
||||||
|
|
||||||
The passed code is evaluated lazily.
|
The passed code is evaluated lazily.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
/+
|
||||||
class ClientSideScript : Element {
|
class ClientSideScript : Element {
|
||||||
private Stack!(string*) codes;
|
private Stack!(string*) codes;
|
||||||
this(Document par) {
|
this(Document par) {
|
||||||
|
@ -993,6 +998,7 @@ class ClientSideScript : Element {
|
||||||
return *v;
|
return *v;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
+/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Interesting things with scripts:
|
Interesting things with scripts:
|
||||||
|
@ -1088,6 +1094,7 @@ import std.stdio;
|
||||||
import std.json;
|
import std.json;
|
||||||
import std.traits;
|
import std.traits;
|
||||||
|
|
||||||
|
/+
|
||||||
string toJavascript(T)(T a) {
|
string toJavascript(T)(T a) {
|
||||||
static if(is(T == ClientSideScript.Variable)) {
|
static if(is(T == ClientSideScript.Variable)) {
|
||||||
return a.name;
|
return a.name;
|
||||||
|
@ -1197,6 +1204,7 @@ string translateJavascriptSourceWithDToStandardScript(string src)() {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
+/
|
+/
|
||||||
|
+/
|
||||||
|
|
||||||
abstract class CssPart {
|
abstract class CssPart {
|
||||||
override string toString() const;
|
override string toString() const;
|
||||||
|
|
6
mysql.d
6
mysql.d
|
@ -691,7 +691,7 @@ string fromCstring(cstring c, int len = -1) {
|
||||||
|
|
||||||
|
|
||||||
// FIXME: this should work generically with all database types and them moved to database.d
|
// FIXME: this should work generically with all database types and them moved to database.d
|
||||||
Ret queryOneRow(Ret = Row, DB, T...)(DB db, string sql, T t) if(
|
Ret queryOneRow(Ret = Row, DB, string file = __FILE__, size_t line = __LINE__, T...)(DB db, string sql, T t) if(
|
||||||
(is(DB : Database))
|
(is(DB : Database))
|
||||||
// && (is(Ret == Row) || is(Ret : DataObject)))
|
// && (is(Ret == Row) || is(Ret : DataObject)))
|
||||||
)
|
)
|
||||||
|
@ -699,12 +699,12 @@ Ret queryOneRow(Ret = Row, DB, T...)(DB db, string sql, T t) if(
|
||||||
static if(is(Ret : DataObject) && is(DB == MySql)) {
|
static if(is(Ret : DataObject) && is(DB == MySql)) {
|
||||||
auto res = db.queryDataObject!Ret(sql, t);
|
auto res = db.queryDataObject!Ret(sql, t);
|
||||||
if(res.empty)
|
if(res.empty)
|
||||||
throw new Exception("result was empty");
|
throw new Exception("result was empty", file, line);
|
||||||
return res.front;
|
return res.front;
|
||||||
} else static if(is(Ret == Row)) {
|
} else static if(is(Ret == Row)) {
|
||||||
auto res = db.query(sql, t);
|
auto res = db.query(sql, t);
|
||||||
if(res.empty)
|
if(res.empty)
|
||||||
throw new Exception("result was empty");
|
throw new Exception("result was empty", file, line);
|
||||||
return res.front;
|
return res.front;
|
||||||
} else static assert(0, "Unsupported single row query return value, " ~ Ret.stringof);
|
} else static assert(0, "Unsupported single row query return value, " ~ Ret.stringof);
|
||||||
}
|
}
|
||||||
|
|
13
oauth.d
13
oauth.d
|
@ -30,12 +30,13 @@ class FacebookApiException : Exception {
|
||||||
|
|
||||||
import arsd.curl;
|
import arsd.curl;
|
||||||
import arsd.sha;
|
import arsd.sha;
|
||||||
import std.md5;
|
import std.digest.md;
|
||||||
|
|
||||||
import std.file;
|
import std.file;
|
||||||
|
|
||||||
|
|
||||||
Variant[string] postToFacebookWall(string[] info, string id, string message, string picture = null, string link = null, long when = 0) {
|
// note when is a d_time, so unix_timestamp * 1000
|
||||||
|
Variant[string] postToFacebookWall(string[] info, string id, string message, string picture = null, string link = null, long when = 0, string linkDescription = null) {
|
||||||
string url = "https://graph.facebook.com/" ~ id ~ "/feed";
|
string url = "https://graph.facebook.com/" ~ id ~ "/feed";
|
||||||
|
|
||||||
|
|
||||||
|
@ -46,8 +47,12 @@ Variant[string] postToFacebookWall(string[] info, string id, string message, str
|
||||||
data ~= "&picture=" ~ std.uri.encodeComponent(picture);
|
data ~= "&picture=" ~ std.uri.encodeComponent(picture);
|
||||||
if(link !is null && link.length)
|
if(link !is null && link.length)
|
||||||
data ~= "&link=" ~ std.uri.encodeComponent(link);
|
data ~= "&link=" ~ std.uri.encodeComponent(link);
|
||||||
if(when)
|
if(when) {
|
||||||
data ~= "&scheduled_publish_time=" ~ to!string(when);
|
data ~= "&scheduled_publish_time=" ~ to!string(when / 1000);
|
||||||
|
data ~= "&published=false";
|
||||||
|
}
|
||||||
|
if(linkDescription.length)
|
||||||
|
data ~= "&description=" ~ std.uri.encodeComponent(linkDescription);
|
||||||
|
|
||||||
auto response = curl(url, data);
|
auto response = curl(url, data);
|
||||||
|
|
||||||
|
|
177
web.d
177
web.d
|
@ -1,5 +1,69 @@
|
||||||
module arsd.web;
|
module arsd.web;
|
||||||
|
|
||||||
|
enum RequirePost;
|
||||||
|
|
||||||
|
/// Attribute for the default formatting (html, table, json, etc)
|
||||||
|
struct DefaultFormat {
|
||||||
|
string format;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Sets the preferred request method, used by things like other code generators.
|
||||||
|
/// While this is preferred, the function is still callable from any request method.
|
||||||
|
///
|
||||||
|
/// By default, the preferred method is GET if the name starts with "get" and POST otherwise.
|
||||||
|
///
|
||||||
|
/// See also: RequirePost, ensureGoodPost, and using Cgi.RequestMethod as an attribute
|
||||||
|
struct PreferredMethod {
|
||||||
|
Cgi.RequestMethod preferredMethod;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// With this attribute, the function is only called if the input data's
|
||||||
|
/// content type is what you specify here. Makes sense for POST and PUT
|
||||||
|
/// verbs.
|
||||||
|
struct IfInputContentType {
|
||||||
|
string contentType;
|
||||||
|
string dataGoesInWhichArgument;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
URL Mapping
|
||||||
|
|
||||||
|
By default, it is the method name OR the method name separated by dashes instead of camel case
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
/+
|
||||||
|
Attributes
|
||||||
|
|
||||||
|
// this is different than calling ensureGoodPost because
|
||||||
|
// it is only called on direct calls. ensureGoodPost is flow oriented
|
||||||
|
enum RequirePost;
|
||||||
|
|
||||||
|
// path info? One could be the name of the current function, one could be the stuff past it...
|
||||||
|
|
||||||
|
// Incomplete form handler
|
||||||
|
|
||||||
|
// overrides the getGenericContainer
|
||||||
|
struct DocumentContainer {}
|
||||||
|
|
||||||
|
// custom formatter for json and other user defined types
|
||||||
|
|
||||||
|
// custom title for the page
|
||||||
|
|
||||||
|
// do we prefill from url? something else? default?
|
||||||
|
struct Prefill {}
|
||||||
|
|
||||||
|
// btw prefill should also take a function
|
||||||
|
// perhaps a FormFinalizer
|
||||||
|
|
||||||
|
// for automatic form creation
|
||||||
|
struct ParameterSuggestions {
|
||||||
|
string[] suggestions;
|
||||||
|
bool showDropdown; /* otherwise it is just autocomplete on a text box */
|
||||||
|
}
|
||||||
|
|
||||||
|
+/
|
||||||
|
|
||||||
// FIXME: if a method has a default value of a non-primitive type,
|
// FIXME: if a method has a default value of a non-primitive type,
|
||||||
// it's still liable to screw everything else.
|
// it's still liable to screw everything else.
|
||||||
|
|
||||||
|
@ -233,7 +297,7 @@ string linkCall(alias Func, Args...)(Args args) {
|
||||||
/// This function works pretty ok. You're going to want to append a string to the return
|
/// This function works pretty ok. You're going to want to append a string to the return
|
||||||
/// value to actually call .get() or whatever; it only does the name and arglist.
|
/// value to actually call .get() or whatever; it only does the name and arglist.
|
||||||
string jsCall(alias Func, Args...)(Args args) /*if(is(__traits(parent, Func) : WebDotDBaseType))*/ {
|
string jsCall(alias Func, Args...)(Args args) /*if(is(__traits(parent, Func) : WebDotDBaseType))*/ {
|
||||||
static if(!__traits(compiles, Func(args))) {
|
static if(!is(typeof(Func(args)))) { //__traits(compiles, Func(args))) {
|
||||||
static assert(0, "Your function call doesn't compile. If you need client side dynamic data, try building the call as a string.");
|
static assert(0, "Your function call doesn't compile. If you need client side dynamic data, try building the call as a string.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -359,10 +423,12 @@ class ApiProvider : WebDotDBaseType {
|
||||||
|
|
||||||
/// we have to add these things to the document...
|
/// we have to add these things to the document...
|
||||||
override void _postProcess(Document document) {
|
override void _postProcess(Document document) {
|
||||||
foreach(pp; documentPostProcessors)
|
if(document !is null) {
|
||||||
pp(document);
|
foreach(pp; documentPostProcessors)
|
||||||
|
pp(document);
|
||||||
|
|
||||||
addCsrfTokens(document);
|
addCsrfTokens(document);
|
||||||
|
}
|
||||||
super._postProcess(document);
|
super._postProcess(document);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -472,12 +538,13 @@ class ApiProvider : WebDotDBaseType {
|
||||||
|
|
||||||
/// This tentatively redirects the user - depends on the envelope fomat
|
/// This tentatively redirects the user - depends on the envelope fomat
|
||||||
/// You can temporarily disable this using disableRedirects()
|
/// You can temporarily disable this using disableRedirects()
|
||||||
void redirect(string location, bool important = false) {
|
string redirect(string location, bool important = false, string status = null) {
|
||||||
if(redirectsSuppressed)
|
if(redirectsSuppressed)
|
||||||
return;
|
return location;
|
||||||
auto f = cgi.request("envelopeFormat", "document");
|
auto f = cgi.request("envelopeFormat", "document");
|
||||||
if(f == "document" || f == "redirect")
|
if(f == "document" || f == "redirect" || f == "json_enable_redirects")
|
||||||
cgi.setResponseLocation(location, important);
|
cgi.setResponseLocation(location, important, status);
|
||||||
|
return location;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns a list of links to all functions in this class or sub-classes
|
/// Returns a list of links to all functions in this class or sub-classes
|
||||||
|
@ -899,6 +966,7 @@ immutable(ReflectionInfo*) prepareReflectionImpl(alias PM, alias Parent)(Parent
|
||||||
reflection.structs[member] = i;
|
reflection.structs[member] = i;
|
||||||
} else static if(
|
} else static if(
|
||||||
is(typeof(__traits(getMember, Class, member)) == function)
|
is(typeof(__traits(getMember, Class, member)) == function)
|
||||||
|
&& __traits(getProtection, __traits(getMember, Class, member)) == "export"
|
||||||
&&
|
&&
|
||||||
(
|
(
|
||||||
member.length < 5 ||
|
member.length < 5 ||
|
||||||
|
@ -1100,9 +1168,10 @@ CallInfo parseUrl(in ReflectionInfo* reflection, string url, string defaultFunct
|
||||||
void run(Provider)(Cgi cgi, Provider instantiation, size_t pathInfoStartingPoint = 0, bool handleAllExceptions = true) if(is(Provider : ApiProvider)) {
|
void run(Provider)(Cgi cgi, Provider instantiation, size_t pathInfoStartingPoint = 0, bool handleAllExceptions = true) if(is(Provider : ApiProvider)) {
|
||||||
assert(instantiation !is null);
|
assert(instantiation !is null);
|
||||||
|
|
||||||
|
instantiation.cgi = cgi;
|
||||||
|
|
||||||
if(instantiation.reflection is null) {
|
if(instantiation.reflection is null) {
|
||||||
instantiation.reflection = prepareReflection!(Provider)(instantiation);
|
instantiation.reflection = prepareReflection!(Provider)(instantiation);
|
||||||
instantiation.cgi = cgi;
|
|
||||||
instantiation._initialize();
|
instantiation._initialize();
|
||||||
// FIXME: what about initializing child objects?
|
// FIXME: what about initializing child objects?
|
||||||
}
|
}
|
||||||
|
@ -1354,6 +1423,7 @@ void run(Provider)(Cgi cgi, Provider instantiation, size_t pathInfoStartingPoint
|
||||||
cgi.setResponseLocation(redirect, false);
|
cgi.setResponseLocation(redirect, false);
|
||||||
break;
|
break;
|
||||||
case "json":
|
case "json":
|
||||||
|
case "json_enable_redirects":
|
||||||
// this makes firefox ugly
|
// this makes firefox ugly
|
||||||
//cgi.setResponseContentType("application/json");
|
//cgi.setResponseContentType("application/json");
|
||||||
auto json = toJsonValue(result);
|
auto json = toJsonValue(result);
|
||||||
|
@ -1452,6 +1522,8 @@ void run(Provider)(Cgi cgi, Provider instantiation, size_t pathInfoStartingPoint
|
||||||
if(realObject !is null)
|
if(realObject !is null)
|
||||||
postProcessors ~= &(realObject._postProcess);
|
postProcessors ~= &(realObject._postProcess);
|
||||||
postProcessors ~= &(base._postProcess);
|
postProcessors ~= &(base._postProcess);
|
||||||
|
|
||||||
|
// FIXME: cgi is sometimes null in te post processor... wtf
|
||||||
foreach(pp; postProcessors) {
|
foreach(pp; postProcessors) {
|
||||||
if(pp in run)
|
if(pp in run)
|
||||||
continue;
|
continue;
|
||||||
|
@ -1975,6 +2047,17 @@ JSONValue toJsonValue(T, R = ApiProvider)(T a, string formatToStringAs = null, R
|
||||||
val.type = JSON_TYPE.STRING;
|
val.type = JSON_TYPE.STRING;
|
||||||
val.str = a.toString();
|
val.str = a.toString();
|
||||||
}
|
}
|
||||||
|
} else static if(is(T == long)) {
|
||||||
|
// FIXME: let's get a better range... I think it goes up to like 1 << 50 on positive and negative
|
||||||
|
// but this works for now
|
||||||
|
if(a < int.max && a > int.min) {
|
||||||
|
val.type = JSON_TYPE.INTEGER;
|
||||||
|
val.integer = to!long(a);
|
||||||
|
} else {
|
||||||
|
// WHY? because javascript can't actually store all 64 bit numbers correctly
|
||||||
|
val.type = JSON_TYPE.STRING;
|
||||||
|
val.str = to!string(a);
|
||||||
|
}
|
||||||
} else static if(isIntegral!(T)) {
|
} else static if(isIntegral!(T)) {
|
||||||
val.type = JSON_TYPE.INTEGER;
|
val.type = JSON_TYPE.INTEGER;
|
||||||
val.integer = to!long(a);
|
val.integer = to!long(a);
|
||||||
|
@ -2530,7 +2613,6 @@ string beautify(string name) {
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
import std.md5;
|
|
||||||
import core.stdc.stdlib;
|
import core.stdc.stdlib;
|
||||||
import core.stdc.time;
|
import core.stdc.time;
|
||||||
import std.file;
|
import std.file;
|
||||||
|
@ -2538,6 +2620,7 @@ import std.file;
|
||||||
/// meant to give a generic useful hook for sessions. kinda sucks at this point.
|
/// meant to give a generic useful hook for sessions. kinda sucks at this point.
|
||||||
/// use the Session class instead. If you just construct it, the sessionId property
|
/// use the Session class instead. If you just construct it, the sessionId property
|
||||||
/// works fine. Don't set any data and it won't save any file.
|
/// works fine. Don't set any data and it won't save any file.
|
||||||
|
version(none)
|
||||||
deprecated string getSessionId(Cgi cgi) {
|
deprecated string getSessionId(Cgi cgi) {
|
||||||
string token; // FIXME: should this actually be static? it seems wrong
|
string token; // FIXME: should this actually be static? it seems wrong
|
||||||
if(token is null) {
|
if(token is null) {
|
||||||
|
@ -2551,6 +2634,7 @@ deprecated string getSessionId(Cgi cgi) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
import std.md5;
|
||||||
return getDigestString(cgi.remoteAddress ~ "\r\n" ~ cgi.userAgent ~ "\r\n" ~ token);
|
return getDigestString(cgi.remoteAddress ~ "\r\n" ~ cgi.userAgent ~ "\r\n" ~ token);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2591,7 +2675,7 @@ class Session {
|
||||||
this._readOnly = readOnly;
|
this._readOnly = readOnly;
|
||||||
|
|
||||||
bool isNew = false;
|
bool isNew = false;
|
||||||
string token;
|
// string token; // using a member, see the note below
|
||||||
if(cookieParams.name in cgi.cookies && cgi.cookies[cookieParams.name].length)
|
if(cookieParams.name in cgi.cookies && cgi.cookies[cookieParams.name].length)
|
||||||
token = cgi.cookies[cookieParams.name];
|
token = cgi.cookies[cookieParams.name];
|
||||||
else {
|
else {
|
||||||
|
@ -2725,7 +2809,10 @@ class Session {
|
||||||
return token;
|
return token;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void setOurCookie(string data) {
|
// FIXME: hack, see note on member string token
|
||||||
|
// don't use this, it is meant to be private (...probably)
|
||||||
|
/*private*/ void setOurCookie(string data) {
|
||||||
|
this.token = data;
|
||||||
if(!_readOnly)
|
if(!_readOnly)
|
||||||
cgi.setCookie(cookieParams.name, data,
|
cgi.setCookie(cookieParams.name, data,
|
||||||
cookieParams.expiresIn, cookieParams.path, cookieParams.host, true, cookieParams.httpsOnly);
|
cookieParams.expiresIn, cookieParams.path, cookieParams.host, true, cookieParams.httpsOnly);
|
||||||
|
@ -2937,6 +3024,8 @@ class Session {
|
||||||
private string _sessionId;
|
private string _sessionId;
|
||||||
private Cgi cgi; // used to regenerate cookies, etc.
|
private Cgi cgi; // used to regenerate cookies, etc.
|
||||||
|
|
||||||
|
string token; // this isn't private, but don't use it FIXME this is a hack to allow cross domain session sharing on the same server....
|
||||||
|
|
||||||
//private Variant[string] data;
|
//private Variant[string] data;
|
||||||
/*
|
/*
|
||||||
Variant* opBinary(string op)(string key) if(op == "in") {
|
Variant* opBinary(string op)(string key) if(op == "in") {
|
||||||
|
@ -2966,6 +3055,7 @@ void setLoginCookie(Cgi cgi, string name, string value) {
|
||||||
|
|
||||||
|
|
||||||
immutable(string[]) monthNames = [
|
immutable(string[]) monthNames = [
|
||||||
|
null,
|
||||||
"January",
|
"January",
|
||||||
"February",
|
"February",
|
||||||
"March",
|
"March",
|
||||||
|
@ -2998,11 +3088,14 @@ struct TemplateFilters {
|
||||||
// string (string replacement, string[], in Element, string)
|
// string (string replacement, string[], in Element, string)
|
||||||
|
|
||||||
string date(string replacement, string[], in Element, string) {
|
string date(string replacement, string[], in Element, string) {
|
||||||
auto dateTicks = to!time_t(replacement);
|
if(replacement.length == 0)
|
||||||
auto date = SysTime( unixTimeToStdTime(dateTicks/1_000) );
|
return replacement;
|
||||||
|
auto dateTicks = to!long(replacement);
|
||||||
|
auto date = SysTime( unixTimeToStdTime(cast(time_t)(dateTicks/1_000)) );
|
||||||
|
|
||||||
auto day = date.day;
|
auto day = date.day;
|
||||||
auto year = date.year;
|
auto year = date.year;
|
||||||
|
assert(date.month < monthNames.length, to!string(date.month));
|
||||||
auto month = monthNames[date.month];
|
auto month = monthNames[date.month];
|
||||||
replacement = format("%s %d, %d", month, day, year);
|
replacement = format("%s %d, %d", month, day, year);
|
||||||
|
|
||||||
|
@ -3079,6 +3172,21 @@ struct TemplateFilters {
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
string stringArray(string replacement, string[] args, in Element, string) {
|
||||||
|
if(replacement.length == 0)
|
||||||
|
return replacement;
|
||||||
|
int idx = to!int(replacement);
|
||||||
|
if(idx < 0 || idx >= args.length)
|
||||||
|
return replacement;
|
||||||
|
return args[idx];
|
||||||
|
}
|
||||||
|
|
||||||
|
string boolean(string replacement, string[] args, in Element, string) {
|
||||||
|
if(replacement == "1")
|
||||||
|
return "yes";
|
||||||
|
return "no";
|
||||||
|
}
|
||||||
|
|
||||||
static auto defaultThings() {
|
static auto defaultThings() {
|
||||||
string delegate(string, string[], in Element, string)[string] pipeFunctions;
|
string delegate(string, string[], in Element, string)[string] pipeFunctions;
|
||||||
TemplateFilters filters;
|
TemplateFilters filters;
|
||||||
|
@ -3421,6 +3529,9 @@ Table structToTable(T)(Document document, T s, string[] fieldsToSkip = null) if(
|
||||||
|
|
||||||
/// This adds a custom attribute to links in the document called qsa which modifies the values on the query string
|
/// This adds a custom attribute to links in the document called qsa which modifies the values on the query string
|
||||||
void translateQsa(Document document, Cgi cgi, string logicalScriptName = null) {
|
void translateQsa(Document document, Cgi cgi, string logicalScriptName = null) {
|
||||||
|
if(document is null || cgi is null)
|
||||||
|
return;
|
||||||
|
|
||||||
if(logicalScriptName is null)
|
if(logicalScriptName is null)
|
||||||
logicalScriptName = cgi.logicalScriptName;
|
logicalScriptName = cgi.logicalScriptName;
|
||||||
|
|
||||||
|
@ -4285,6 +4396,44 @@ enum string javascriptBaseImpl = q{
|
||||||
"_wantScriptExecution" : true,
|
"_wantScriptExecution" : true,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
template hasAnnotation(alias f, Attr) {
|
||||||
|
bool helper() {
|
||||||
|
foreach(attr; __traits(getAttributes, f))
|
||||||
|
static if(is(attr == Attr) || is(typeof(attr) == Attr))
|
||||||
|
return true;
|
||||||
|
return false;
|
||||||
|
|
||||||
|
}
|
||||||
|
enum bool hasAnnotation = helper;
|
||||||
|
}
|
||||||
|
|
||||||
|
template hasValueAnnotation(alias f, Attr) {
|
||||||
|
bool helper() {
|
||||||
|
foreach(attr; __traits(getAttributes, f))
|
||||||
|
static if(is(typeof(attr) == Attr))
|
||||||
|
return true;
|
||||||
|
return false;
|
||||||
|
|
||||||
|
}
|
||||||
|
enum bool hasValueAnnotation = helper;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
template getAnnotation(alias f, Attr) if(hasValueAnnotation!(f, Attr)) {
|
||||||
|
auto helper() {
|
||||||
|
foreach(attr; __traits(getAttributes, f))
|
||||||
|
static if(is(typeof(attr) == Attr))
|
||||||
|
return attr;
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
enum getAnnotation = helper;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright: Adam D. Ruppe, 2010 - 2012
|
Copyright: Adam D. Ruppe, 2010 - 2012
|
||||||
License: <a href="http://www.boost.org/LICENSE_1_0.txt">Boost License 1.0</a>.
|
License: <a href="http://www.boost.org/LICENSE_1_0.txt">Boost License 1.0</a>.
|
||||||
|
|
Loading…
Reference in New Issue