diff --git a/README.md b/README.md
index 7114bf6..fbf3e23 100644
--- a/README.md
+++ b/README.md
@@ -200,3 +200,7 @@ There's a few other hidden gems in the files themselves, and so much more on my
(Yes, I'm writing a pair of D games again, finally! First time in a long time, but it is moving along well... and I don't need SDL this time!)
+
+# Special Conventions
+
+idl Starting in 2019, I will be adding version info to individual modules.
diff --git a/cgi.d b/cgi.d
index d4e44c5..d05c100 100644
--- a/cgi.d
+++ b/cgi.d
@@ -1,6 +1,8 @@
// FIXME: if an exception is thrown, we shouldn't necessarily cache...
// FIXME: there's some annoying duplication of code in the various versioned mains
+// FIXME: cgi per-request arena allocator
+
// FIXME: I might make a cgi proxy class which can change things; the underlying one is still immutable
// but the later one can edit and simplify the api. You'd have to use the subclass tho!
@@ -621,9 +623,6 @@ class Cgi {
environmentVariables = cast(const) environment.toAA;
- string[] allPostNamesInOrder;
- string[] allPostValuesInOrder;
-
foreach(arg; args[1 .. $]) {
if(arg.startsWith("--")) {
nextArgIs = arg[2 .. $];
@@ -3757,8 +3756,13 @@ class ListeningConnectionManager {
// certainly does for small requests, and I think it does for larger ones too
sn.setOption(SocketOptionLevel.TCP, SocketOption.TCP_NODELAY, 1);
}
- while(queueLength >= queue.length)
+ // wait until a slot opens up
+ //int waited = 0;
+ while(queueLength >= queue.length) {
Thread.sleep(1.msecs);
+ //waited ++;
+ }
+ //if(waited) {import std.stdio; writeln(waited);}
synchronized(this) {
queue[nextIndexBack] = sn;
nextIndexBack++;
@@ -3815,6 +3819,7 @@ class ListeningConnectionManager {
tcp = true;
}
+ Thread.getThis.priority = Thread.PRIORITY_MAX;
listener.listen(128);
}
@@ -5069,7 +5074,7 @@ class Session(Data) {
cgi.setCookie(
"sessionId", sessionId,
0 /* expiration */,
- null /* path */,
+ "/" /* path */,
null /* domain */,
true /* http only */,
cgi.https /* if the session is started on https, keep it there, otherwise, be flexible */);
@@ -5913,24 +5918,21 @@ ssize_t read_fd(int fd, void *ptr, size_t nbytes, int *recvfd) {
switch to choose if you want to override.
*/
-struct DispatcherDefinition(alias dispatchHandler) {// if(is(typeof(dispatchHandler("str", Cgi.init, void) == bool))) { // bool delegate(string urlPrefix, Cgi cgi) dispatchHandler;
+struct DispatcherDefinition(alias dispatchHandler, DispatcherDetails = typeof(null)) {// if(is(typeof(dispatchHandler("str", Cgi.init, void) == bool))) { // bool delegate(string urlPrefix, Cgi cgi) dispatchHandler;
alias handler = dispatchHandler;
string urlPrefix;
bool rejectFurther;
- DispatcherDetails details;
-}
-
-// tbh I am really unhappy with this part
-struct DispatcherDetails {
- string filename;
- string contentType;
+ immutable(DispatcherDetails) details;
}
private string urlify(string name) {
- return name;
+ return beautify(name, '-', true);
}
-private string beautify(string name) {
+private string beautify(string name, char space = ' ', bool allLowerCase = false) {
+ if(name == "id")
+ return allLowerCase ? name : "ID";
+
char[160] buffer;
int bufferIndex = 0;
bool shouldCap = true;
@@ -5939,7 +5941,7 @@ private string beautify(string name) {
foreach(idx, char ch; name) {
if(bufferIndex == buffer.length) return name; // out of space, just give up, not that important
- if(ch >= 'A' && ch <= 'Z') {
+ if((ch >= 'A' && ch <= 'Z') || ch == '_') {
if(lastWasCap) {
// two caps in a row, don't change. Prolly acronym.
} else {
@@ -5951,206 +5953,22 @@ private string beautify(string name) {
}
if(shouldSpace) {
- buffer[bufferIndex++] = ' ';
+ buffer[bufferIndex++] = space;
if(bufferIndex == buffer.length) return name; // out of space, just give up, not that important
+ shouldSpace = false;
}
if(shouldCap) {
if(ch >= 'a' && ch <= 'z')
ch -= 32;
shouldCap = false;
}
+ if(allLowerCase && ch >= 'A' && ch <= 'Z')
+ ch += 32;
buffer[bufferIndex++] = ch;
}
return buffer[0 .. bufferIndex].idup;
}
-/+
- Argument conversions: for the most part, it is to!Thing(string).
-
- But arrays and structs are a bit different. Arrays come from the cgi array. Thus
- they are passed
-
- arr=foo&arr=bar <-- notice the same name.
-
- Structs are first declared with an empty thing, then have their members set individually,
- with dot notation. The members are not required, just the initial declaration.
-
- struct Foo {
- int a;
- string b;
- }
- void test(Foo foo){}
-
- foo&foo.a=5&foo.b=str <-- the first foo declares the arg, the others set the members
-
- Arrays of structs use this declaration.
-
- void test(Foo[] foo) {}
-
- foo&foo.a=5&foo.b=bar&foo&foo.a=9
-
- You can use a hidden input field in HTML forms to achieve this. The value of the naked name
- declaration is ignored.
-
- Mind that order matters! The declaration MUST come first in the string.
-
- Arrays of struct members follow this rule recursively.
-
- struct Foo {
- int[] a;
- }
-
- foo&foo.a=1&foo.a=2&foo&foo.a=1
-
-
- Associative arrays are formatted with brackets, after a declaration, like structs:
-
- foo&foo[key]=value&foo[other_key]=value
-
-
- Note: for maximum compatibility with outside code, keep your types simple. Some libraries
- do not support the strict ordering requirements to work with these struct protocols.
-
- FIXME: also perhaps accept application/json to better work with outside trash.
-
-
- Return values are also auto-formatted according to user-requested type:
- for json, it loops over and converts.
- for html, basic types are strings. Arrays are
. Structs are
. Arrays of structs are tables!
-+/
-
-// returns an arsd.dom.Element
-static auto elementFor(T)(string displayName, string name) {
- import arsd.dom;
- import std.traits;
-
- auto div = Element.make("div");
- div.addClass("form-field");
-
- static if(is(T == struct)) {
- if(displayName !is null)
- div.addChild("span", displayName, "label-text");
- auto fieldset = div.addChild("fieldset");
- fieldset.addChild("legend", beautify(T.stringof)); // FIXME
- fieldset.addChild("input", name);
- static foreach(idx, memberName; __traits(allMembers, T))
- static if(__traits(compiles, __traits(getMember, T, memberName).offsetof)) {
- fieldset.appendChild(elementFor!(typeof(__traits(getMember, T, memberName)))(beautify(memberName), name ~ "." ~ memberName));
- }
- } else static if(isSomeString!T || isIntegral!T || isFloatingPoint!T) {
- Element lbl;
- if(displayName !is null) {
- lbl = div.addChild("label");
- lbl.addChild("span", displayName, "label-text");
- lbl.appendText(" ");
- } else {
- lbl = div;
- }
- auto i = lbl.addChild("input", name);
- i.attrs.name = name;
- static if(isSomeString!T)
- i.attrs.type = "text";
- else
- i.attrs.type = "number";
- i.attrs.value = to!string(T.init);
- } else static if(is(T == bool)) {
- Element lbl;
- if(displayName !is null) {
- lbl = div.addChild("label");
- lbl.addChild("span", displayName, "label-text");
- lbl.appendText(" ");
- } else {
- lbl = div;
- }
- auto i = lbl.addChild("input", name);
- i.attrs.type = "checkbox";
- i.attrs.name = name;
- } else static if(is(T == K[], K)) {
- auto templ = div.addChild("template");
- templ.appendChild(elementFor!(K)(null, name));
- if(displayName !is null)
- div.addChild("span", displayName, "label-text");
- auto btn = div.addChild("button");
- btn.addClass("add-array-button");
- btn.attrs.type = "button";
- btn.innerText = "Add";
- btn.attrs.onclick = q{
- var a = document.importNode(this.parentNode.firstChild.content, true);
- this.parentNode.insertBefore(a, this);
- };
- } else static if(is(T == V[K], K, V)) {
- div.innerText = "assoc array not implemented for automatic form at this time";
- } else {
- static assert(0, "unsupported type for cgi call " ~ T.stringof);
- }
-
-
- return div;
-}
-
-// actually returns an arsd.dom.Form
-auto createAutomaticFormForFunction(alias method, T)(T dg) {
- import arsd.dom;
-
- auto form = cast(Form) Element.make("form");
-
- form.addClass("automatic-form");
-
- form.addChild("h3", beautify(__traits(identifier, method)));
-
- import std.traits;
-
- //Parameters!method params;
- //alias idents = ParameterIdentifierTuple!method;
- //alias defaults = ParameterDefaults!method;
-
- static if(is(typeof(method) P == __parameters))
- static foreach(idx, _; P) {{
- alias param = P[idx .. idx + 1];
- string displayName = beautify(__traits(identifier, param));
- static foreach(attr; __traits(getAttributes, param))
- static if(is(typeof(attr) == DisplayName))
- displayName = attr.name;
- form.appendChild(elementFor!(param)(displayName, __traits(identifier, param)));
- }}
-
- form.addChild("div", Html(``), "submit-button-holder");
-
- return form;
-}
-
-// actually returns an arsd.dom.Form
-auto createAutomaticFormForObject(T)(T obj) {
- import arsd.dom;
-
- auto form = cast(Form) Element.make("form");
-
- form.addClass("automatic-form");
-
- form.addChild("h3", beautify(__traits(identifier, T)));
-
- import std.traits;
-
- //Parameters!method params;
- //alias idents = ParameterIdentifierTuple!method;
- //alias defaults = ParameterDefaults!method;
-
- static foreach(idx, memberName; __traits(derivedMembers, T)) {{
- static if(__traits(compiles, __traits(getMember, obj, memberName).offsetof)) {
- string displayName = beautify(memberName);
- static foreach(attr; __traits(getAttributes, __traits(getMember, T, memberName)))
- static if(is(typeof(attr) == DisplayName))
- displayName = attr.name;
- form.appendChild(elementFor!(typeof(__traits(getMember, T, memberName)))(displayName, memberName));
-
- form.setValue(memberName, to!string(__traits(getMember, obj, memberName)));
- }}}
-
- form.addChild("div", Html(``), "submit-button-holder");
-
- return form;
-}
-
/*
string urlFor(alias func)() {
return __traits(identifier, func);
@@ -6189,6 +6007,7 @@ class MissingArgumentException : Exception {
}
}
+// it only looks at query params for GET requests, the rest must be in the body for a function argument.
auto callFromCgi(alias method, T)(T dg, Cgi cgi) {
// FIXME: any array of structs should also be settable or gettable from csv as well.
@@ -6206,20 +6025,24 @@ auto callFromCgi(alias method, T)(T dg, Cgi cgi) {
// first, check for missing arguments and initialize to defaults if necessary
static foreach(idx, param; params) {{
- auto ident = idents[idx];
- if(cgi.requestMethod == Cgi.RequestMethod.POST) {
- if(ident !in cgi.post) {
- static if(is(defaults[idx] == void))
- throw new MissingArgumentException(__traits(identifier, method), ident, typeof(param).stringof);
- else
- params[idx] = defaults[idx];
- }
+ static if(is(typeof(param) : Cgi)) {
+ params[idx] = cgi;
} else {
- if(ident !in cgi.get) {
- static if(is(defaults[idx] == void))
- throw new MissingArgumentException(__traits(identifier, method), ident, typeof(param).stringof);
- else
- params[idx] = defaults[idx];
+ auto ident = idents[idx];
+ if(cgi.requestMethod == Cgi.RequestMethod.GET) {
+ if(ident !in cgi.get) {
+ static if(is(defaults[idx] == void))
+ throw new MissingArgumentException(__traits(identifier, method), ident, typeof(param).stringof);
+ else
+ params[idx] = defaults[idx];
+ }
+ } else {
+ if(ident !in cgi.post) {
+ static if(is(defaults[idx] == void))
+ throw new MissingArgumentException(__traits(identifier, method), ident, typeof(param).stringof);
+ else
+ params[idx] = defaults[idx];
+ }
}
}
}}
@@ -6330,21 +6153,25 @@ auto callFromCgi(alias method, T)(T dg, Cgi cgi) {
sw: switch(paramName) {
static foreach(idx, param; params) {
- case idents[idx]:
- setVariable(name, paramName, ¶ms[idx], value);
- break sw;
+ static if(is(typeof(param) : Cgi)) {
+ // cannot be set from the outside
+ } else {
+ case idents[idx]:
+ setVariable(name, paramName, ¶ms[idx], value);
+ break sw;
+ }
}
default:
// ignore; not relevant argument
}
}
- if(cgi.requestMethod == Cgi.RequestMethod.POST) {
- names = cgi.allPostNamesInOrder;
- values = cgi.allPostValuesInOrder;
- } else {
+ if(cgi.requestMethod == Cgi.RequestMethod.GET) {
names = cgi.allGetNamesInOrder;
values = cgi.allGetValuesInOrder;
+ } else {
+ names = cgi.allPostNamesInOrder;
+ values = cgi.allPostValuesInOrder;
}
foreach(idx, name; names) {
@@ -6364,94 +6191,60 @@ auto callFromCgi(alias method, T)(T dg, Cgi cgi) {
return ret;
}
-auto formatReturnValueAsHtml(T)(T t) {
- import arsd.dom;
- import std.traits;
+/+
+ Argument conversions: for the most part, it is to!Thing(string).
- static if(is(T == typeof(null))) {
- return Element.make("span");
- } else static if(isIntegral!T || isSomeString!T || isFloatingPoint!T) {
- return Element.make("span", to!string(t), "automatic-data-display");
- } else static if(is(T == V[K], K, V)) {
- auto dl = Element.make("dl");
- dl.addClass("automatic-data-display");
- foreach(k, v; t) {
- dl.addChild("dt", to!string(k));
- dl.addChild("dd", formatReturnValueAsHtml(v));
- }
- return dl;
- } else static if(is(T == struct)) {
- auto dl = Element.make("dl");
- dl.addClass("automatic-data-display");
+ But arrays and structs are a bit different. Arrays come from the cgi array. Thus
+ they are passed
- static foreach(idx, memberName; __traits(allMembers, T))
- static if(__traits(compiles, __traits(getMember, T, memberName).offsetof)) {
- dl.addChild("dt", memberName);
- dl.addChild("dt", formatReturnValueAsHtml(__traits(getMember, t, memberName)));
- }
+ arr=foo&arr=bar <-- notice the same name.
- return dl;
- } else static if(is(T == bool)) {
- return Element.make("span", t ? "true" : "false", "automatic-data-display");
- } else static if(is(T == E[], E)) {
- static if(is(E : RestObject!Proxy, Proxy)) {
- // treat RestObject similar to struct
- auto table = cast(Table) Element.make("table");
- table.addClass("automatic-data-display");
- string[] names;
- static foreach(idx, memberName; __traits(derivedMembers, E))
- static if(__traits(compiles, __traits(getMember, E, memberName).offsetof)) {
- names ~= beautify(memberName);
- }
- table.appendHeaderRow(names);
+ Structs are first declared with an empty thing, then have their members set individually,
+ with dot notation. The members are not required, just the initial declaration.
- foreach(l; t) {
- auto tr = table.appendRow();
- static foreach(idx, memberName; __traits(derivedMembers, E))
- static if(__traits(compiles, __traits(getMember, E, memberName).offsetof)) {
- static if(memberName == "id") {
- string val = to!string(__traits(getMember, l, memberName));
- tr.addChild("td", Element.make("a", val, E.stringof.toLower ~ "s/" ~ val)); // FIXME
- } else {
- tr.addChild("td", formatReturnValueAsHtml(__traits(getMember, l, memberName)));
- }
- }
- }
+ struct Foo {
+ int a;
+ string b;
+ }
+ void test(Foo foo){}
- return table;
- } else static if(is(E == struct)) {
- // an array of structs is kinda special in that I like
- // having those formatted as tables.
- auto table = cast(Table) Element.make("table");
- table.addClass("automatic-data-display");
- string[] names;
- static foreach(idx, memberName; __traits(allMembers, E))
- static if(__traits(compiles, __traits(getMember, E, memberName).offsetof)) {
- names ~= beautify(memberName);
- }
- table.appendHeaderRow(names);
+ foo&foo.a=5&foo.b=str <-- the first foo declares the arg, the others set the members
- foreach(l; t) {
- auto tr = table.appendRow();
- static foreach(idx, memberName; __traits(allMembers, E))
- static if(__traits(compiles, __traits(getMember, E, memberName).offsetof)) {
- tr.addChild("td", formatReturnValueAsHtml(__traits(getMember, l, memberName)));
- }
- }
+ Arrays of structs use this declaration.
- return table;
- } else {
- // otherwise, I will just make a list.
- auto ol = Element.make("ol");
- ol.addClass("automatic-data-display");
- foreach(e; t)
- ol.addChild("li", formatReturnValueAsHtml(e));
- return ol;
- }
- } else static assert(0, "bad return value for cgi call " ~ T.stringof);
+ void test(Foo[] foo) {}
- assert(0);
-}
+ foo&foo.a=5&foo.b=bar&foo&foo.a=9
+
+ You can use a hidden input field in HTML forms to achieve this. The value of the naked name
+ declaration is ignored.
+
+ Mind that order matters! The declaration MUST come first in the string.
+
+ Arrays of struct members follow this rule recursively.
+
+ struct Foo {
+ int[] a;
+ }
+
+ foo&foo.a=1&foo.a=2&foo&foo.a=1
+
+
+ Associative arrays are formatted with brackets, after a declaration, like structs:
+
+ foo&foo[key]=value&foo[other_key]=value
+
+
+ Note: for maximum compatibility with outside code, keep your types simple. Some libraries
+ do not support the strict ordering requirements to work with these struct protocols.
+
+ FIXME: also perhaps accept application/json to better work with outside trash.
+
+
+ Return values are also auto-formatted according to user-requested type:
+ for json, it loops over and converts.
+ for html, basic types are strings. Arrays are . Structs are
. Arrays of structs are tables!
++/
/++
A web presenter is responsible for rendering things to HTML to be usable
@@ -6459,9 +6252,27 @@ auto formatReturnValueAsHtml(T)(T t) {
They are passed as template arguments to the base classes of [WebObject]
- FIXME
+ Responsible for displaying stuff as HTML. You can put this into your own aggregate
+ and override it. Use forwarding and specialization to customize it.
+
+ When you inherit from it, pass your own class as the CRTP argument. This lets the base
+ class templates and your overridden templates work with each other.
+
+ ---
+ class MyPresenter : WebPresenter!(MyPresenter) {
+ void presentSuccessfulReturnAsHtml(T : CustomType)(Cgi cgi, T ret) {
+ // present the CustomType
+ }
+ void presentSuccessfulReturnAsHtml(T)(Cgi cgi, T ret) {
+ // handle everything else via the super class, which will call
+ // back to your class when appropriate
+ super.presentSuccessfulReturnAsHtml(cgi, ret);
+ }
+ }
+ ---
+
+/
-class WebPresenter() {
+class WebPresenter(CRTP) {
string script() {
return `
`;
@@ -6578,27 +6389,431 @@ class WebPresenter() {
return document.requireElementById("container");
}
+
+ /// typeof(null) (which is also used to represent functions returning `void`) do nothing
+ /// in the default presenter - allowing the function to have full low-level control over the
+ /// response.
+ void presentSuccessfulReturnAsHtml(T : typeof(null))(Cgi cgi, T ret) {
+ // nothing intentionally!
+ }
+
+ /// Redirections are forwarded to [Cgi.setResponseLocation]
+ void presentSuccessfulReturnAsHtml(T : Redirection)(Cgi cgi, T ret) {
+ cgi.setResponseLocation(ret.to, true, getHttpCodeText(ret.code));
+ }
+
+ /// Multiple responses deconstruct the algebraic type and forward to the appropriate handler at runtime
+ void presentSuccessfulReturnAsHtml(T : MultipleResponses!Types, Types...)(Cgi cgi, T ret) {
+ bool outputted = false;
+ static foreach(index, type; Types) {
+ if(ret.contains == index) {
+ assert(!outputted);
+ outputted = true;
+ (cast(CRTP) this).presentSuccessfulReturnAsHtml(cgi, ret.payload[index]);
+ }
+ }
+ if(!outputted)
+ assert(0);
+ }
+
+ /// And the default handler will call [formatReturnValueAsHtml] and place it inside the [htmlContainer].
+ void presentSuccessfulReturnAsHtml(T)(Cgi cgi, T ret) {
+ auto container = this.htmlContainer();
+ container.appendChild(formatReturnValueAsHtml(ret));
+ cgi.write(container.parentDocument.toString(), true);
+ }
+
+ /++
+ If you override this, you will need to cast the exception type `t` dynamically,
+ but can then use the template arguments here to refer back to the function.
+
+ `func` is an alias to the method itself, and `dg` is a callable delegate to the same
+ method on the live object. You could, in theory, change arguments and retry, but I
+ provide that information mostly with the expectation that you will use them to make
+ useful forms or richer error messages for the user.
+ +/
+ void presentExceptionAsHtml(alias func, T)(Cgi cgi, Throwable t, T dg) {
+ if(auto mae = cast(MissingArgumentException) t) {
+ auto container = this.htmlContainer();
+ container.appendChild(Element.make("p", "Argument `" ~ mae.argumentName ~ "` of type `" ~ mae.argumentType ~ "` is missing"));
+ container.appendChild(createAutomaticFormForFunction!(func)(dg));
+
+ cgi.write(container.parentDocument.toString(), true);
+ } else {
+ auto container = this.htmlContainer();
+
+ // import std.stdio; writeln(t.toString());
+
+ container.addChild("pre", t.toString());
+
+ cgi.setResponseStatus("500 Internal Server Error");
+ cgi.write(container.parentDocument.toString(), true);
+ }
+ }
+
+ /++
+ returns an arsd.dom.Element
+ +/
+ auto elementFor(T)(string displayName, string name) {
+ import arsd.dom;
+ import std.traits;
+
+ auto div = Element.make("div");
+ div.addClass("form-field");
+
+ static if(is(T == struct)) {
+ if(displayName !is null)
+ div.addChild("span", displayName, "label-text");
+ auto fieldset = div.addChild("fieldset");
+ fieldset.addChild("legend", beautify(T.stringof)); // FIXME
+ fieldset.addChild("input", name);
+ static foreach(idx, memberName; __traits(allMembers, T))
+ static if(__traits(compiles, __traits(getMember, T, memberName).offsetof)) {
+ fieldset.appendChild(elementFor!(typeof(__traits(getMember, T, memberName)))(beautify(memberName), name ~ "." ~ memberName));
+ }
+ } else static if(isSomeString!T || isIntegral!T || isFloatingPoint!T) {
+ Element lbl;
+ if(displayName !is null) {
+ lbl = div.addChild("label");
+ lbl.addChild("span", displayName, "label-text");
+ lbl.appendText(" ");
+ } else {
+ lbl = div;
+ }
+ auto i = lbl.addChild("input", name);
+ i.attrs.name = name;
+ static if(isSomeString!T)
+ i.attrs.type = "text";
+ else
+ i.attrs.type = "number";
+ i.attrs.value = to!string(T.init);
+ } else static if(is(T == bool)) {
+ Element lbl;
+ if(displayName !is null) {
+ lbl = div.addChild("label");
+ lbl.addChild("span", displayName, "label-text");
+ lbl.appendText(" ");
+ } else {
+ lbl = div;
+ }
+ auto i = lbl.addChild("input", name);
+ i.attrs.type = "checkbox";
+ i.attrs.name = name;
+ } else static if(is(T == K[], K)) {
+ auto templ = div.addChild("template");
+ templ.appendChild(elementFor!(K)(null, name));
+ if(displayName !is null)
+ div.addChild("span", displayName, "label-text");
+ auto btn = div.addChild("button");
+ btn.addClass("add-array-button");
+ btn.attrs.type = "button";
+ btn.innerText = "Add";
+ btn.attrs.onclick = q{
+ var a = document.importNode(this.parentNode.firstChild.content, true);
+ this.parentNode.insertBefore(a, this);
+ };
+ } else static if(is(T == V[K], K, V)) {
+ div.innerText = "assoc array not implemented for automatic form at this time";
+ } else {
+ static assert(0, "unsupported type for cgi call " ~ T.stringof);
+ }
+
+
+ return div;
+ }
+
+ /// actually returns an arsd.dom.Form
+ auto createAutomaticFormForFunction(alias method, T)(T dg) {
+ import arsd.dom;
+
+ auto form = cast(Form) Element.make("form");
+
+ form.addClass("automatic-form");
+
+ form.addChild("h3", beautify(__traits(identifier, method)));
+
+ import std.traits;
+
+ //Parameters!method params;
+ //alias idents = ParameterIdentifierTuple!method;
+ //alias defaults = ParameterDefaults!method;
+
+ static if(is(typeof(method) P == __parameters))
+ static foreach(idx, _; P) {{
+ static if(!is(_ : Cgi)) {
+ alias param = P[idx .. idx + 1];
+ string displayName = beautify(__traits(identifier, param));
+ static foreach(attr; __traits(getAttributes, param))
+ static if(is(typeof(attr) == DisplayName))
+ displayName = attr.name;
+ form.appendChild(elementFor!(param)(displayName, __traits(identifier, param)));
+ }
+ }}
+
+ form.addChild("div", Html(``), "submit-button-holder");
+
+ return form;
+ }
+
+ /// actually returns an arsd.dom.Form
+ auto createAutomaticFormForObject(T)(T obj) {
+ import arsd.dom;
+
+ auto form = cast(Form) Element.make("form");
+
+ form.addClass("automatic-form");
+
+ form.addChild("h3", beautify(__traits(identifier, T)));
+
+ import std.traits;
+
+ //Parameters!method params;
+ //alias idents = ParameterIdentifierTuple!method;
+ //alias defaults = ParameterDefaults!method;
+
+ static foreach(idx, memberName; __traits(derivedMembers, T)) {{
+ static if(__traits(compiles, __traits(getMember, obj, memberName).offsetof)) {
+ string displayName = beautify(memberName);
+ static foreach(attr; __traits(getAttributes, __traits(getMember, T, memberName)))
+ static if(is(typeof(attr) == DisplayName))
+ displayName = attr.name;
+ form.appendChild(elementFor!(typeof(__traits(getMember, T, memberName)))(displayName, memberName));
+
+ form.setValue(memberName, to!string(__traits(getMember, obj, memberName)));
+ }}}
+
+ form.addChild("div", Html(``), "submit-button-holder");
+
+ return form;
+ }
+
+ ///
+ auto formatReturnValueAsHtml(T)(T t) {
+ import arsd.dom;
+ import std.traits;
+
+ static if(is(T == typeof(null))) {
+ return Element.make("span");
+ } else static if(is(T == MultipleResponses!Types, Types...)) {
+ static foreach(index, type; Types) {
+ if(t.contains == index)
+ return formatReturnValueAsHtml(t.payload[index]);
+ }
+ assert(0);
+ } else static if(isIntegral!T || isSomeString!T || isFloatingPoint!T) {
+ return Element.make("span", to!string(t), "automatic-data-display");
+ } else static if(is(T == V[K], K, V)) {
+ auto dl = Element.make("dl");
+ dl.addClass("automatic-data-display");
+ foreach(k, v; t) {
+ dl.addChild("dt", to!string(k));
+ dl.addChild("dd", formatReturnValueAsHtml(v));
+ }
+ return dl;
+ } else static if(is(T == struct)) {
+ auto dl = Element.make("dl");
+ dl.addClass("automatic-data-display");
+
+ static foreach(idx, memberName; __traits(allMembers, T))
+ static if(__traits(compiles, __traits(getMember, T, memberName).offsetof)) {
+ dl.addChild("dt", memberName);
+ dl.addChild("dt", formatReturnValueAsHtml(__traits(getMember, t, memberName)));
+ }
+
+ return dl;
+ } else static if(is(T == bool)) {
+ return Element.make("span", t ? "true" : "false", "automatic-data-display");
+ } else static if(is(T == E[], E)) {
+ static if(is(E : RestObject!Proxy, Proxy)) {
+ // treat RestObject similar to struct
+ auto table = cast(Table) Element.make("table");
+ table.addClass("automatic-data-display");
+ string[] names;
+ static foreach(idx, memberName; __traits(derivedMembers, E))
+ static if(__traits(compiles, __traits(getMember, E, memberName).offsetof)) {
+ names ~= beautify(memberName);
+ }
+ table.appendHeaderRow(names);
+
+ foreach(l; t) {
+ auto tr = table.appendRow();
+ static foreach(idx, memberName; __traits(derivedMembers, E))
+ static if(__traits(compiles, __traits(getMember, E, memberName).offsetof)) {
+ static if(memberName == "id") {
+ string val = to!string(__traits(getMember, l, memberName));
+ tr.addChild("td", Element.make("a", val, E.stringof.toLower ~ "s/" ~ val)); // FIXME
+ } else {
+ tr.addChild("td", formatReturnValueAsHtml(__traits(getMember, l, memberName)));
+ }
+ }
+ }
+
+ return table;
+ } else static if(is(E == struct)) {
+ // an array of structs is kinda special in that I like
+ // having those formatted as tables.
+ auto table = cast(Table) Element.make("table");
+ table.addClass("automatic-data-display");
+ string[] names;
+ static foreach(idx, memberName; __traits(allMembers, E))
+ static if(__traits(compiles, __traits(getMember, E, memberName).offsetof)) {
+ names ~= beautify(memberName);
+ }
+ table.appendHeaderRow(names);
+
+ foreach(l; t) {
+ auto tr = table.appendRow();
+ static foreach(idx, memberName; __traits(allMembers, E))
+ static if(__traits(compiles, __traits(getMember, E, memberName).offsetof)) {
+ tr.addChild("td", formatReturnValueAsHtml(__traits(getMember, l, memberName)));
+ }
+ }
+
+ return table;
+ } else {
+ // otherwise, I will just make a list.
+ auto ol = Element.make("ol");
+ ol.addClass("automatic-data-display");
+ foreach(e; t)
+ ol.addChild("li", formatReturnValueAsHtml(e));
+ return ol;
+ }
+ } else static assert(0, "bad return value for cgi call " ~ T.stringof);
+
+ assert(0);
+ }
+
}
/++
The base class for the [dispatcher] function and object support.
+/
-class WebObject(Helper = void) {
- Cgi cgi;
- WebPresenter!() presenter;
+class WebObject {
+ //protected Cgi cgi;
- void initialize(Cgi cgi, WebPresenter!() presenter) {
- this.cgi = cgi;
- this.presenter = presenter;
+ protected void initialize(Cgi cgi) {
+ //this.cgi = cgi;
}
}
+/++
+ Can return one of the given types, decided at runtime. The syntax
+ is to declare all the possible types in the return value, then you
+ can `return typeof(return)(...value...)` to construct it.
+
+ It has an auto-generated constructor for each value it can hold.
+
+ ---
+ MultipleResponses!(Redirection, string) getData(int how) {
+ if(how & 1)
+ return typeof(return)(Redirection("http://dpldocs.info/"));
+ else
+ return typeof(return)("hi there!");
+ }
+ ---
+
+ If you have lots of returns, you could, inside the function, `alias r = typeof(return);` to shorten it a little.
++/
+struct MultipleResponses(T...) {
+ private size_t contains;
+ private union {
+ private T payload;
+ }
+
+ static foreach(index, type; T)
+ public this(type t) {
+ contains = index;
+ payload[index] = t;
+ }
+
+ /++
+ This is primarily for testing. It is your way of getting to the response.
+
+ Let's say you wanted to test that one holding a Redirection and a string actually
+ holds a string, by name of "test":
+
+ ---
+ auto valueToTest = your_test_function();
+
+ valueToTest.visit!(
+ (Redirection) { assert(0); }, // got a redirection instead of a string, fail the test
+ (string s) { assert(s == "test"); } // right value, go ahead and test it.
+ );
+ ---
+ +/
+ void visit(Handlers...)() {
+ template findHandler(type, HandlersToCheck...) {
+ static if(HandlersToCheck.length == 0)
+ alias findHandler = void;
+ else {
+ static if(is(typeof(HandlersToCheck[0](type.init))))
+ alias findHandler = handler;
+ else
+ alias findHandler = findHandler!(type, HandlersToCheck[1 .. $]);
+ }
+ }
+ static foreach(index, type; T) {{
+ alias handler = findHandler!(type, Handlers);
+ static if(is(handler == void))
+ static assert(0, "Type " ~ type.stringof ~ " was not handled by visitor");
+ else {
+ if(index == contains)
+ handler(payload[index]);
+ }
+ }}
+ }
+
+ /+
+ auto toArsdJsvar()() {
+ import arsd.jsvar;
+ return var(null);
+ }
+ +/
+}
+
+struct RawResponse {
+ int code;
+ string[] headers;
+ const(ubyte)[] responseBody;
+}
+
+/++
+ You can return this from [WebObject] subclasses for redirections.
+
+ (though note the static types means that class must ALWAYS redirect if
+ you return this directly. You might want to return [MultipleResponses] if it
+ can be conditional)
++/
+struct Redirection {
+ string to; /// The URL to redirect to.
+ int code = 303; /// The HTTP code to retrn.
+}
+
/++
Serves a class' methods, as a kind of low-state RPC over the web. To be used with [dispatcher].
- Usage of this function will add a dependency on [arsd.dom] and [arsd.jsvar].
+ Usage of this function will add a dependency on [arsd.dom] and [arsd.jsvar] unless you have overriden
+ the presenter in the dispatcher.
FIXME: explain this better
+
+ You can overload functions to a limited extent: you can provide a zero-arg and non-zero-arg function,
+ and non-zero-arg functions can filter via UDAs for various http methods. Do not attempt other overloads,
+ the runtime result of that is undefined.
+
+ A method is assumed to allow any http method unless it lists some in UDAs, in which case it is limited to only those.
+ (this might change, like maybe i will use pure as an indicator GET is ok. idk.)
+
+ $(WARNING
+ ---
+ // legal in D, undefined runtime behavior with cgi.d, it may call either method
+ // even if you put different URL udas on it, the current code ignores them.
+ void foo(int a) {}
+ void foo(string a) {}
+ ---
+ )
+
+ See_Also: [serveRestObject], [serveStaticFile]
+/
auto serveApi(T)(string urlPrefix) {
assert(urlPrefix[$ - 1] == '/');
@@ -6606,53 +6821,134 @@ auto serveApi(T)(string urlPrefix) {
import arsd.dom;
import arsd.jsvar;
- static bool handler(string urlPrefix, Cgi cgi, WebPresenter!() presenter, DispatcherDetails details) {
+ static bool internalHandler(Presenter)(string urlPrefix, Cgi cgi, Presenter presenter, immutable void* details) {
auto obj = new T();
- obj.initialize(cgi, presenter);
+ obj.initialize(cgi);
switch(cgi.pathInfo[urlPrefix.length .. $]) {
static foreach(methodName; __traits(derivedMembers, T)){{
static if(is(typeof(__traits(getMember, T, methodName)) P == __parameters))
{
- case urlify(methodName):
+ case urlNameForMethod!(__traits(getMember, T, methodName))(urlify(methodName)):
+ int zeroArgOverload = -1;
+ int overloadCount = cast(int) __traits(getOverloads, T, methodName).length;
+ bool calledWithZeroArgs = true;
+ foreach(k, v; cgi.get)
+ if(k != "format") {
+ calledWithZeroArgs = false;
+ break;
+ }
+ foreach(k, v; cgi.post)
+ if(k != "format") {
+ calledWithZeroArgs = false;
+ break;
+ }
+
+ // first, we need to go through and see if there is an empty one, since that
+ // changes inside. But otherwise, all the stuff I care about can be done via
+ // simple looping (other improper overloads might be flagged for runtime semantic check)
+ //
+ // an argument of type Cgi is ignored for these purposes
+ static foreach(idx, overload; __traits(getOverloads, T, methodName)) {{
+ static if(is(typeof(overload) P == __parameters))
+ static if(P.length == 0)
+ zeroArgOverload = cast(int) idx;
+ else static if(P.length == 1 && is(P[0] : Cgi))
+ zeroArgOverload = cast(int) idx;
+ }}
+ // FIXME: static assert if there are multiple non-zero-arg overloads usable with a single http method.
+ bool overloadHasBeenCalled = false;
+ static foreach(idx, overload; __traits(getOverloads, T, methodName)) {{
+ bool callFunction = true;
+ // there is a zero arg overload and this is NOT it, and we have zero args - don't call this
+ if(overloadCount > 1 && zeroArgOverload != -1 && idx != zeroArgOverload && calledWithZeroArgs)
+ callFunction = false;
+ // if this is the zero-arg overload, obviously it cannot be called if we got any args.
+ if(overloadCount > 1 && idx == zeroArgOverload && !calledWithZeroArgs)
+ callFunction = false;
+
+ // FIXME: so if you just add ?foo it will give the error below even when. this might not be a great idea.
+
+ bool hadAnyMethodRestrictions = false;
+ bool foundAcceptableMethod = false;
+ foreach(attr; __traits(getAttributes, overload)) {
+ static if(is(typeof(attr) == Cgi.RequestMethod)) {
+ hadAnyMethodRestrictions = true;
+ if(attr == cgi.requestMethod)
+ foundAcceptableMethod = true;
+ }
+ }
+
+ if(hadAnyMethodRestrictions && !foundAcceptableMethod)
+ callFunction = false;
+
+ /+
+ The overloads we really want to allow are the sane ones
+ from the web perspective. Which is likely on HTTP verbs,
+ for the most part, but might also be potentially based on
+ some args vs zero args, or on argument names. Can't really
+ do argument types very reliable through the web though; those
+ should probably be different URLs.
+
+ Even names I feel is better done inside the function, so I'm not
+ going to support that here. But the HTTP verbs and zero vs some
+ args makes sense - it lets you define custom forms pretty easily.
+
+ Moreover, I'm of the opinion that empty overload really only makes
+ sense on GET for this case. On a POST, it is just a missing argument
+ exception and that should be handled by the presenter. But meh, I'll
+ let the user define that, D only allows one empty arg thing anyway
+ so the method UDAs are irrelevant.
+ +/
+ if(callFunction)
switch(cgi.request("format", "html")) {
case "html":
- auto container = obj.presenter.htmlContainer();
try {
- auto ret = callFromCgi!(__traits(getMember, obj, methodName))(&__traits(getMember, obj, methodName), cgi);
- container.appendChild(formatReturnValueAsHtml(ret));
- } catch(MissingArgumentException mae) {
- container.appendChild(Element.make("p", "Argument `" ~ mae.argumentName ~ "` of type `" ~ mae.argumentType ~ "` is missing"));
- container.appendChild(createAutomaticFormForFunction!(__traits(getMember, obj, methodName))(&__traits(getMember, obj, methodName)));
+ // a void return (or typeof(null) lol) means you, the user, is doing it yourself. Gives full control.
+ auto ret = callFromCgi!(__traits(getOverloads, obj, methodName)[idx])(&(__traits(getOverloads, obj, methodName)[idx]), cgi);
+ presenter.presentSuccessfulReturnAsHtml(cgi, ret);
+ } catch(Throwable t) {
+ presenter.presentExceptionAsHtml!(__traits(getOverloads, obj, methodName)[idx])(cgi, t, &(__traits(getOverloads, obj, methodName)[idx]));
}
- cgi.write(container.parentDocument.toString(), true);
- break;
+ return true;
case "json":
- auto ret = callFromCgi!(__traits(getMember, obj, methodName))(&__traits(getMember, obj, methodName), cgi);
- var json = ret;
+ auto ret = callFromCgi!(__traits(getOverloads, obj, methodName)[idx])(&(__traits(getOverloads, obj, methodName)[idx]), cgi);
+ static if(is(typeof(ret) == MultipleResponses!Types, Types...)) {
+ var json;
+ static foreach(index, type; Types) {
+ if(ret.contains == index)
+ json = ret.payload[index];
+ }
+ } else {
+ var json = ret;
+ }
var envelope = var.emptyObject;
envelope.success = true;
envelope.result = json;
envelope.error = null;
cgi.setResponseContentType("application/json");
cgi.write(envelope.toJson(), true);
-
- break;
+ return true;
default:
+ cgi.setResponseStatus("406 Not Acceptable"); // not exactly but sort of.
+ return true;
}
+ }}
+ cgi.header("Accept: POST"); // FIXME list the real thing
+ cgi.setResponseStatus("405 Method Not Allowed"); // again, not exactly, but sort of. no overload matched our args, almost certainly due to http verb filtering.
return true;
}
}}
case "script.js":
cgi.setResponseContentType("text/javascript");
cgi.gzipResponse = true;
- cgi.write(obj.presenter.script(), true);
+ cgi.write(presenter.script(), true);
return true;
case "style.css":
cgi.setResponseContentType("text/css");
cgi.gzipResponse = true;
- cgi.write(obj.presenter.style(), true);
+ cgi.write(presenter.style(), true);
return true;
default:
return false;
@@ -6660,7 +6956,15 @@ auto serveApi(T)(string urlPrefix) {
assert(0);
}
- return DispatcherDefinition!handler(urlPrefix, false);
+ return DispatcherDefinition!internalHandler(urlPrefix, false);
+}
+
+string urlNameForMethod(alias method)(string def) {
+ static foreach(attr; __traits(getAttributes, method)) {
+ static if(is(typeof(attr) == UrlName))
+ return attr.name;
+ }
+ return def;
}
@@ -6695,7 +6999,7 @@ auto serveApi(T)(string urlPrefix) {
/++
The base of all REST objects, to be used with [serveRestObject] and [serveRestCollectionOf].
+/
-class RestObject(Helper = void) : WebObject!Helper {
+class RestObject(CRTP) : WebObject {
import arsd.dom;
import arsd.jsvar;
@@ -6784,21 +7088,13 @@ class RestObject(Helper = void) : WebObject!Helper {
+/
}
-/++
- Responsible for displaying stuff as HTML. You can put this into your own aggregate
- and override it. Use forwarding and specialization to customize it.
-+/
-mixin template Presenter() {
-
-}
-
// FIXME XSRF token, prolly can just put in a cookie and then it needs to be copied to header or form hidden value
// https://use-the-index-luke.com/sql/partial-results/fetch-next-page
/++
Base class for REST collections.
+/
-class CollectionOf(Obj, Helper = void) : RestObject!(Helper) {
+class CollectionOf(Obj) : RestObject!(CollectionOf) {
/// You might subclass this and use the cgi object's query params
/// to implement a search filter, for example.
///
@@ -6960,8 +7256,9 @@ class CollectionOf(Obj, Helper = void) : RestObject!(Helper) {
+/
auto serveRestObject(T)(string urlPrefix) {
+ assert(urlPrefix[0] == '/');
assert(urlPrefix[$ - 1] != '/', "Do NOT use a trailing slash on REST objects.");
- static bool handler(string urlPrefix, Cgi cgi, WebPresenter!() presenter, DispatcherDetails details) {
+ static bool internalHandler(Presenter)(string urlPrefix, Cgi cgi, Presenter presenter, immutable void* details) {
string url = cgi.pathInfo[urlPrefix.length .. $];
if(url.length && url[$ - 1] == '/') {
@@ -6971,20 +7268,22 @@ auto serveRestObject(T)(string urlPrefix) {
}
return restObjectServeHandler!T(cgi, presenter, url);
-
}
- return DispatcherDefinition!handler(urlPrefix, false);
+ return DispatcherDefinition!internalHandler(urlPrefix, false);
}
+/+
/// Convenience method for serving a collection. It will be named the same
/// as type T, just with an s at the end. If you need any further, just
/// write the class yourself.
auto serveRestCollectionOf(T)(string urlPrefix) {
+ assert(urlPrefix[0] == '/');
mixin(`static class `~T.stringof~`s : CollectionOf!(T) {}`);
return serveRestObject!(mixin(T.stringof ~ "s"))(urlPrefix);
}
++/
-bool restObjectServeHandler(T)(Cgi cgi, WebPresenter!() presenter, string url) {
+bool restObjectServeHandler(T, Presenter)(Cgi cgi, Presenter presenter, string url) {
string urlId = null;
if(url.length && url[0] == '/') {
// asking for a subobject
@@ -6999,9 +7298,9 @@ bool restObjectServeHandler(T)(Cgi cgi, WebPresenter!() presenter, string url) {
// FIXME handle other subresources
- static if(is(T : CollectionOf!(C, P), C, P)) {
+ static if(is(T : CollectionOf!(C), C)) {
if(urlId !is null) {
- return restObjectServeHandler!C(cgi, presenter, url); // FIXME? urlId);
+ return restObjectServeHandler!(C, Presenter)(cgi, presenter, url); // FIXME? urlId);
}
}
@@ -7017,7 +7316,7 @@ bool restObjectServeHandler(T)(Cgi cgi, WebPresenter!() presenter, string url) {
static foreach(idx, memberName; __traits(derivedMembers, T))
static if(__traits(compiles, __traits(getMember, obj, memberName).offsetof)) {
if(!first) div.addChild("br"); else first = false;
- div.appendChild(formatReturnValueAsHtml(__traits(getMember, obj, memberName)));
+ div.appendChild(presenter.formatReturnValueAsHtml(__traits(getMember, obj, memberName)));
}
return div;
};
@@ -7034,7 +7333,7 @@ bool restObjectServeHandler(T)(Cgi cgi, WebPresenter!() presenter, string url) {
// FIXME
return ValidationResult.valid;
};
- obj.initialize(cgi, presenter);
+ obj.initialize(cgi);
// FIXME: populate reflection info delegates
@@ -7043,12 +7342,12 @@ bool restObjectServeHandler(T)(Cgi cgi, WebPresenter!() presenter, string url) {
case "script.js":
cgi.setResponseContentType("text/javascript");
cgi.gzipResponse = true;
- cgi.write(obj.presenter.script(), true);
+ cgi.write(presenter.script(), true);
return true;
case "style.css":
cgi.setResponseContentType("text/css");
cgi.gzipResponse = true;
- cgi.write(obj.presenter.style(), true);
+ cgi.write(presenter.style(), true);
return true;
default:
// intentionally blank
@@ -7074,9 +7373,9 @@ bool restObjectServeHandler(T)(Cgi cgi, WebPresenter!() presenter, string url) {
cgi.setResponseContentType("application/json");
cgi.write(obj.toJson().toString, true);
} else {
- auto container = obj.presenter.htmlContainer();
+ auto container = presenter.htmlContainer();
if(addFormLinks) {
- static if(is(T : CollectionOf!(C, P), C, P))
+ static if(is(T : CollectionOf!(C), C))
container.appendHtml(`