mirror of https://github.com/adamdruppe/arsd.git
catching up on random stuff
This commit is contained in:
parent
4ff49846d4
commit
79b2a25b92
4
cgi.d
4
cgi.d
|
@ -1163,10 +1163,12 @@ class Cgi {
|
||||||
|
|
||||||
/// This gets a full url for the current request, including port, protocol, host, path, and query
|
/// This gets a full url for the current request, including port, protocol, host, path, and query
|
||||||
string getCurrentCompleteUri() const {
|
string getCurrentCompleteUri() const {
|
||||||
|
ushort defaultPort = https ? 443 : 80;
|
||||||
|
|
||||||
return format("http%s://%s%s%s",
|
return format("http%s://%s%s%s",
|
||||||
https ? "s" : "",
|
https ? "s" : "",
|
||||||
host,
|
host,
|
||||||
port == 80 ? "" : ":" ~ to!string(port),
|
port == defaultPort ? "" : ":" ~ to!string(port),
|
||||||
requestUri);
|
requestUri);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
24
dom.d
24
dom.d
|
@ -20,6 +20,9 @@
|
||||||
*/
|
*/
|
||||||
module arsd.dom;
|
module arsd.dom;
|
||||||
|
|
||||||
|
// FIXME: something like <ol>spam <ol> with no closing </ol> should read the second tag as the closer in garbage mode
|
||||||
|
// FIXME: failing to close a paragraph sometimes messes things up too
|
||||||
|
|
||||||
// FIXME: it would be kinda cool to have some support for internal DTDs
|
// FIXME: it would be kinda cool to have some support for internal DTDs
|
||||||
// and maybe XPath as well, to some extent
|
// and maybe XPath as well, to some extent
|
||||||
/*
|
/*
|
||||||
|
@ -1583,7 +1586,7 @@ class Element {
|
||||||
/**
|
/**
|
||||||
Takes some html and replaces the element's children with the tree made from the string.
|
Takes some html and replaces the element's children with the tree made from the string.
|
||||||
*/
|
*/
|
||||||
Element innerHTML(string html) {
|
Element innerHTML(string html, bool strict = false) {
|
||||||
if(html.length)
|
if(html.length)
|
||||||
selfClosed = false;
|
selfClosed = false;
|
||||||
|
|
||||||
|
@ -1595,7 +1598,7 @@ class Element {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto doc = new Document();
|
auto doc = new Document();
|
||||||
doc.parse("<innerhtml>" ~ html ~ "</innerhtml>"); // FIXME: this should preserve the strictness of the parent document
|
doc.parse("<innerhtml>" ~ html ~ "</innerhtml>", strict, strict); // FIXME: this should preserve the strictness of the parent document
|
||||||
|
|
||||||
children = doc.root.children;
|
children = doc.root.children;
|
||||||
foreach(c; children) {
|
foreach(c; children) {
|
||||||
|
@ -3029,7 +3032,7 @@ class TableCell : Element {
|
||||||
|
|
||||||
|
|
||||||
///.
|
///.
|
||||||
class MarkupError : Exception {
|
class MarkupException : Exception {
|
||||||
|
|
||||||
///.
|
///.
|
||||||
this(string message) {
|
this(string message) {
|
||||||
|
@ -3208,7 +3211,7 @@ class Document : FileResource {
|
||||||
|
|
||||||
if(dataEncoding is null) {
|
if(dataEncoding is null) {
|
||||||
if(strict)
|
if(strict)
|
||||||
throw new MarkupError("I couldn't figure out the encoding of this document.");
|
throw new MarkupException("I couldn't figure out the encoding of this document.");
|
||||||
else
|
else
|
||||||
// if we really don't know by here, it means we already tried UTF-8,
|
// if we really don't know by here, it means we already tried UTF-8,
|
||||||
// looked for utf 16 and 32 byte order marks, and looked for xml or meta
|
// looked for utf 16 and 32 byte order marks, and looked for xml or meta
|
||||||
|
@ -3282,7 +3285,7 @@ class Document : FileResource {
|
||||||
}
|
}
|
||||||
|
|
||||||
void parseError(string message) {
|
void parseError(string message) {
|
||||||
throw new MarkupError(format("char %d (line %d): %s", pos, getLineNumber(pos), message));
|
throw new MarkupException(format("char %d (line %d): %s", pos, getLineNumber(pos), message));
|
||||||
}
|
}
|
||||||
|
|
||||||
void eatWhitespace() {
|
void eatWhitespace() {
|
||||||
|
@ -3312,7 +3315,7 @@ class Document : FileResource {
|
||||||
data[pos] != ' ' && data[pos] != '\n' && data[pos] != '\t')
|
data[pos] != ' ' && data[pos] != '\n' && data[pos] != '\t')
|
||||||
{
|
{
|
||||||
if(data[pos] == '<')
|
if(data[pos] == '<')
|
||||||
throw new MarkupError("The character < can never appear in an attribute name.");
|
throw new MarkupException("The character < can never appear in an attribute name.");
|
||||||
pos++;
|
pos++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3326,11 +3329,14 @@ class Document : FileResource {
|
||||||
switch(data[pos]) {
|
switch(data[pos]) {
|
||||||
case '\'':
|
case '\'':
|
||||||
case '"':
|
case '"':
|
||||||
|
auto started = pos;
|
||||||
char end = data[pos];
|
char end = data[pos];
|
||||||
pos++;
|
pos++;
|
||||||
auto start = pos;
|
auto start = pos;
|
||||||
while(data[pos] != end)
|
while(pos < data.length && data[pos] != end)
|
||||||
pos++;
|
pos++;
|
||||||
|
if(strict && pos == data.length)
|
||||||
|
throw new MarkupException("Unclosed attribute value, started on char " ~ to!string(started));
|
||||||
string v = htmlEntitiesDecode(data[start..pos], strict);
|
string v = htmlEntitiesDecode(data[start..pos], strict);
|
||||||
pos++; // skip over the end
|
pos++; // skip over the end
|
||||||
return v;
|
return v;
|
||||||
|
@ -3387,7 +3393,7 @@ class Document : FileResource {
|
||||||
if(pos >= data.length)
|
if(pos >= data.length)
|
||||||
{
|
{
|
||||||
if(strict) {
|
if(strict) {
|
||||||
throw new MarkupError("Gone over the input (is there no root element?), chain: " ~ to!string(parentChain));
|
throw new MarkupException("Gone over the input (is there no root element?), chain: " ~ to!string(parentChain));
|
||||||
} else {
|
} else {
|
||||||
if(parentChain.length)
|
if(parentChain.length)
|
||||||
return Ele(1, null, parentChain[0]); // in loose mode, we just assume the document has ended
|
return Ele(1, null, parentChain[0]); // in loose mode, we just assume the document has ended
|
||||||
|
@ -3672,7 +3678,7 @@ class Document : FileResource {
|
||||||
}
|
}
|
||||||
|
|
||||||
if(strict && attrName in attributes)
|
if(strict && attrName in attributes)
|
||||||
throw new MarkupError("Repeated attribute: " ~ attrName);
|
throw new MarkupException("Repeated attribute: " ~ attrName);
|
||||||
attributes[attrName] = attrValue;
|
attributes[attrName] = attrValue;
|
||||||
|
|
||||||
goto moreAttributes;
|
goto moreAttributes;
|
||||||
|
|
43
engine.d
43
engine.d
|
@ -8,6 +8,13 @@
|
||||||
*/
|
*/
|
||||||
module arsd.engine; //@-L-lSDL -L-lSDL_mixer -L-lSDL_ttf -L-lSDL_image -L-lGL -L-lSDL_net
|
module arsd.engine; //@-L-lSDL -L-lSDL_mixer -L-lSDL_ttf -L-lSDL_image -L-lGL -L-lSDL_net
|
||||||
|
|
||||||
|
pragma(lib, "SDL");
|
||||||
|
pragma(lib, "SDL_mixer");
|
||||||
|
pragma(lib, "SDL_ttf");
|
||||||
|
pragma(lib, "SDL_image");
|
||||||
|
pragma(lib, "SDL_net");
|
||||||
|
pragma(lib, "GL");
|
||||||
|
|
||||||
// FIXME: the difference between directions and buttons should be removed
|
// FIXME: the difference between directions and buttons should be removed
|
||||||
|
|
||||||
|
|
||||||
|
@ -45,8 +52,6 @@ else
|
||||||
import std.stdio;
|
import std.stdio;
|
||||||
//version(linux) pragma(lib, "kbhit.o");
|
//version(linux) pragma(lib, "kbhit.o");
|
||||||
|
|
||||||
extern(C) bool kbhit();
|
|
||||||
|
|
||||||
int randomNumber(int min, int max){
|
int randomNumber(int min, int max){
|
||||||
if(min == max)
|
if(min == max)
|
||||||
return min;
|
return min;
|
||||||
|
@ -1211,3 +1216,37 @@ bool directionIsDown(Engine.Direction d, int which = 0){
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
version(linux) {
|
||||||
|
version(D_Version2) {
|
||||||
|
import sys = core.sys.posix.sys.select;
|
||||||
|
version=CustomKbhit;
|
||||||
|
|
||||||
|
int kbhit()
|
||||||
|
{
|
||||||
|
sys.timeval tv;
|
||||||
|
sys.fd_set read_fd;
|
||||||
|
|
||||||
|
tv.tv_sec=0;
|
||||||
|
tv.tv_usec=0;
|
||||||
|
sys.FD_ZERO(&read_fd);
|
||||||
|
sys.FD_SET(0,&read_fd);
|
||||||
|
|
||||||
|
if(sys.select(1, &read_fd, null, null, &tv) == -1)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if(sys.FD_ISSET(0,&read_fd))
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// else, use kbhit.o from the C file
|
||||||
|
}
|
||||||
|
|
||||||
|
version(CustomKbhit) {} else
|
||||||
|
extern(C) bool kbhit();
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
15
htmltotext.d
15
htmltotext.d
|
@ -21,8 +21,8 @@ string htmlToText(string html, bool wantWordWrap = true, int wrapAmount = 74) {
|
||||||
html = html.replace(" ", " ");
|
html = html.replace(" ", " ");
|
||||||
html = html.replace(" ", " ");
|
html = html.replace(" ", " ");
|
||||||
html = html.replace(" ", " ");
|
html = html.replace(" ", " ");
|
||||||
html = html.replace("\n", "");
|
html = html.replace("\n", " ");
|
||||||
html = html.replace("\r", "");
|
html = html.replace("\r", " ");
|
||||||
html = std.regex.replace(html, std.regex.regex("[\n\r\t \u00a0]+", "gm"), " ");
|
html = std.regex.replace(html, std.regex.regex("[\n\r\t \u00a0]+", "gm"), " ");
|
||||||
|
|
||||||
document.parse("<roottag>" ~ html ~ "</roottag>");
|
document.parse("<roottag>" ~ html ~ "</roottag>");
|
||||||
|
@ -75,6 +75,7 @@ string htmlToText(string html, bool wantWordWrap = true, int wrapAmount = 74) {
|
||||||
ele.stripOut();
|
ele.stripOut();
|
||||||
goto again;
|
goto again;
|
||||||
break;
|
break;
|
||||||
|
case "td":
|
||||||
case "p":
|
case "p":
|
||||||
/*
|
/*
|
||||||
if(ele.innerHTML.length > 1)
|
if(ele.innerHTML.length > 1)
|
||||||
|
@ -121,7 +122,14 @@ string htmlToText(string html, bool wantWordWrap = true, int wrapAmount = 74) {
|
||||||
start.innerHTML = start.innerHTML().replace("\u0001", "\n");
|
start.innerHTML = start.innerHTML().replace("\u0001", "\n");
|
||||||
|
|
||||||
foreach(ele; start.tree) {
|
foreach(ele; start.tree) {
|
||||||
if(ele.tagName == "p") {
|
if(ele.tagName == "td") {
|
||||||
|
if(ele.directText().strip().length) {
|
||||||
|
ele.prependText("\r");
|
||||||
|
ele.appendText("\r");
|
||||||
|
}
|
||||||
|
ele.stripOut();
|
||||||
|
goto again2;
|
||||||
|
} else if(ele.tagName == "p") {
|
||||||
if(strip(ele.innerText()).length > 1) {
|
if(strip(ele.innerText()).length > 1) {
|
||||||
string res = "";
|
string res = "";
|
||||||
string all = ele.innerText().replace("\n \n", "\n\n");
|
string all = ele.innerText().replace("\n \n", "\n\n");
|
||||||
|
@ -136,6 +144,7 @@ string htmlToText(string html, bool wantWordWrap = true, int wrapAmount = 74) {
|
||||||
}
|
}
|
||||||
|
|
||||||
result = start.innerText();
|
result = start.innerText();
|
||||||
|
result = squeeze(result, " ");
|
||||||
|
|
||||||
result = result.replace("\r ", "\r");
|
result = result.replace("\r ", "\r");
|
||||||
result = result.replace(" \r", "\r");
|
result = result.replace(" \r", "\r");
|
||||||
|
|
16
jpg.d
16
jpg.d
|
@ -5,7 +5,6 @@ import std.stdio;
|
||||||
import std.conv;
|
import std.conv;
|
||||||
|
|
||||||
struct JpegSection {
|
struct JpegSection {
|
||||||
ushort length;
|
|
||||||
ubyte identifier;
|
ubyte identifier;
|
||||||
ubyte[] data;
|
ubyte[] data;
|
||||||
}
|
}
|
||||||
|
@ -37,16 +36,16 @@ struct LazyJpegFile {
|
||||||
throw new Exception("not lined up in file");
|
throw new Exception("not lined up in file");
|
||||||
|
|
||||||
_front.identifier = startingBuffer[1];
|
_front.identifier = startingBuffer[1];
|
||||||
_front.length = cast(ushort) (startingBuffer[2]) * 256 + startingBuffer[3];
|
ushort length = cast(ushort) (startingBuffer[2]) * 256 + startingBuffer[3];
|
||||||
|
|
||||||
if(_front.length < 2)
|
if(length < 2)
|
||||||
throw new Exception("wtf");
|
throw new Exception("wtf");
|
||||||
_front.length -= 2; // the length in the file includes the block header, but we just want the data here
|
length -= 2; // the length in the file includes the block header, but we just want the data here
|
||||||
|
|
||||||
_front.data = new ubyte[](_front.length);
|
_front.data = new ubyte[](length);
|
||||||
read = f.rawRead(_front.data);
|
read = f.rawRead(_front.data);
|
||||||
if(read.length != _front.length)
|
if(read.length != length)
|
||||||
throw new Exception("didn't read the file right, got " ~ to!string(read.length) ~ " instead of " ~ to!string(_front.length));
|
throw new Exception("didn't read the file right, got " ~ to!string(read.length) ~ " instead of " ~ to!string(length));
|
||||||
|
|
||||||
_frontIsValid = true;
|
_frontIsValid = true;
|
||||||
}
|
}
|
||||||
|
@ -60,7 +59,7 @@ struct LazyJpegFile {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// http://www.obrador.com/essentialjpeg/headerinfo.htm
|
// returns width, height
|
||||||
Tuple!(int, int) getSizeFromFile(string filename) {
|
Tuple!(int, int) getSizeFromFile(string filename) {
|
||||||
import std.stdio;
|
import std.stdio;
|
||||||
|
|
||||||
|
@ -71,6 +70,7 @@ Tuple!(int, int) getSizeFromFile(string filename) {
|
||||||
auto firstSection = jpeg.front();
|
auto firstSection = jpeg.front();
|
||||||
jpeg.popFront();
|
jpeg.popFront();
|
||||||
|
|
||||||
|
// commented because exif and jfif are both readable by this so no need to be picky
|
||||||
//if(firstSection.identifier != 0xe0)
|
//if(firstSection.identifier != 0xe0)
|
||||||
//throw new Exception("bad header");
|
//throw new Exception("bad header");
|
||||||
|
|
||||||
|
|
10
mysql.d
10
mysql.d
|
@ -699,16 +699,22 @@ Ret queryOneRow(Ret = Row, DB, string file = __FILE__, size_t line = __LINE__, T
|
||||||
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", file, line);
|
throw new EmptyResultException("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", file, line);
|
throw new EmptyResultException("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);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class EmptyResultException : Exception {
|
||||||
|
this(string message, string file = __FILE__, size_t line = __LINE__) {
|
||||||
|
super(message, file, line);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
void main() {
|
void main() {
|
||||||
|
|
35
png.d
35
png.d
|
@ -230,6 +230,41 @@ ubyte[] writePng(PNG* p) {
|
||||||
return a;
|
return a;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
PNGHeader getHeaderFromFile(string filename) {
|
||||||
|
import std.stdio;
|
||||||
|
auto file = File(filename, "rb");
|
||||||
|
ubyte[12] initialBuffer; // file header + size of first chunk (should be IHDR)
|
||||||
|
auto data = file.rawRead(initialBuffer[]);
|
||||||
|
if(data.length != 12)
|
||||||
|
throw new Exception("couldn't get png file header off " ~ filename);
|
||||||
|
|
||||||
|
if(data[0..8] != [0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a])
|
||||||
|
throw new Exception("file " ~ filename ~ " is not a png");
|
||||||
|
|
||||||
|
auto pos = 8;
|
||||||
|
uint size;
|
||||||
|
size |= data[pos++] << 24;
|
||||||
|
size |= data[pos++] << 16;
|
||||||
|
size |= data[pos++] << 8;
|
||||||
|
size |= data[pos++] << 0;
|
||||||
|
|
||||||
|
size += 4; // chunk type
|
||||||
|
size += 4; // checksum
|
||||||
|
|
||||||
|
ubyte[] more;
|
||||||
|
more.length = size;
|
||||||
|
|
||||||
|
auto chunk = file.rawRead(more);
|
||||||
|
if(chunk.length != size)
|
||||||
|
throw new Exception("couldn't get png image header off " ~ filename);
|
||||||
|
|
||||||
|
|
||||||
|
more = data ~ chunk;
|
||||||
|
|
||||||
|
auto png = readPng(more);
|
||||||
|
return getHeader(png);
|
||||||
|
}
|
||||||
|
|
||||||
PNG* readPng(ubyte[] data) {
|
PNG* readPng(ubyte[] data) {
|
||||||
auto p = new PNG;
|
auto p = new PNG;
|
||||||
|
|
||||||
|
|
226
web.d
226
web.d
|
@ -1,6 +1,8 @@
|
||||||
module arsd.web;
|
module arsd.web;
|
||||||
|
|
||||||
enum RequirePost;
|
enum RequirePost;
|
||||||
|
enum RequireHttps;
|
||||||
|
enum NoAutomaticForm;
|
||||||
|
|
||||||
/// Attribute for the default formatting (html, table, json, etc)
|
/// Attribute for the default formatting (html, table, json, etc)
|
||||||
struct DefaultFormat {
|
struct DefaultFormat {
|
||||||
|
@ -624,7 +626,10 @@ class ApiProvider : WebDotDBaseType {
|
||||||
document.cookie = \"timezone=\" + tz + \"; path=/\";
|
document.cookie = \"timezone=\" + tz + \"; path=/\";
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
<style>.format-row { display: none; }</style>
|
<style>
|
||||||
|
.format-row { display: none; }
|
||||||
|
.validation-failed { background-color: #ffe0e0; }
|
||||||
|
</style>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<div id=\"body\"></div>
|
<div id=\"body\"></div>
|
||||||
|
@ -784,6 +789,8 @@ struct FunctionInfo {
|
||||||
bool returnTypeIsDocument; // internal used when wrapping
|
bool returnTypeIsDocument; // internal used when wrapping
|
||||||
bool returnTypeIsElement; // internal used when wrapping
|
bool returnTypeIsElement; // internal used when wrapping
|
||||||
|
|
||||||
|
bool requireHttps;
|
||||||
|
|
||||||
Document delegate(in string[string] args) createForm; /// This is used if you want a custom form - normally, on insufficient parameters, an automatic form is created. But if there's a functionName_Form method, it is used instead. FIXME: this used to work but not sure if it still does
|
Document delegate(in string[string] args) createForm; /// This is used if you want a custom form - normally, on insufficient parameters, an automatic form is created. But if there's a functionName_Form method, it is used instead. FIXME: this used to work but not sure if it still does
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -978,6 +985,7 @@ immutable(ReflectionInfo*) prepareReflectionImpl(alias PM, alias Parent)(Parent
|
||||||
FunctionInfo* f = new FunctionInfo;
|
FunctionInfo* f = new FunctionInfo;
|
||||||
ParameterTypeTuple!(__traits(getMember, Class, member)) fargs;
|
ParameterTypeTuple!(__traits(getMember, Class, member)) fargs;
|
||||||
|
|
||||||
|
f.requireHttps = hasAnnotation!(__traits(getMember, Class, member), RequireHttps);
|
||||||
f.returnType = ReturnType!(__traits(getMember, Class, member)).stringof;
|
f.returnType = ReturnType!(__traits(getMember, Class, member)).stringof;
|
||||||
f.returnTypeIsDocument = is(ReturnType!(__traits(getMember, Class, member)) : Document);
|
f.returnTypeIsDocument = is(ReturnType!(__traits(getMember, Class, member)) : Document);
|
||||||
f.returnTypeIsElement = is(ReturnType!(__traits(getMember, Class, member)) : Element);
|
f.returnTypeIsElement = is(ReturnType!(__traits(getMember, Class, member)) : Element);
|
||||||
|
@ -1248,6 +1256,10 @@ void run(Provider)(Cgi cgi, Provider instantiation, size_t pathInfoStartingPoint
|
||||||
}
|
}
|
||||||
|
|
||||||
bool returnedHoldsADocument = false;
|
bool returnedHoldsADocument = false;
|
||||||
|
string[][string] want;
|
||||||
|
string format, secondaryFormat;
|
||||||
|
void delegate(Document d) moreProcessing;
|
||||||
|
WrapperReturn ret;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
if(fun is null) {
|
if(fun is null) {
|
||||||
|
@ -1276,6 +1288,12 @@ void run(Provider)(Cgi cgi, Provider instantiation, size_t pathInfoStartingPoint
|
||||||
assert(fun.dispatcher !is null);
|
assert(fun.dispatcher !is null);
|
||||||
assert(cgi !is null);
|
assert(cgi !is null);
|
||||||
|
|
||||||
|
if(fun.requireHttps && !cgi.https) {
|
||||||
|
cgi.setResponseLocation("https://" ~ cgi.host ~ cgi.logicalScriptName ~ cgi.pathInfo ~
|
||||||
|
(cgi.queryString.length ? "?" : "") ~ cgi.queryString);
|
||||||
|
envelopeFormat = "no-processing";
|
||||||
|
goto do_nothing_else;
|
||||||
|
}
|
||||||
|
|
||||||
if(instantiator.length) {
|
if(instantiator.length) {
|
||||||
assert(fun !is null);
|
assert(fun !is null);
|
||||||
|
@ -1287,14 +1305,14 @@ void run(Provider)(Cgi cgi, Provider instantiation, size_t pathInfoStartingPoint
|
||||||
|
|
||||||
result.type = fun.returnType;
|
result.type = fun.returnType;
|
||||||
|
|
||||||
string format = cgi.request("format", reflection.defaultOutputFormat);
|
format = cgi.request("format", reflection.defaultOutputFormat);
|
||||||
string secondaryFormat = cgi.request("secondaryFormat", "");
|
secondaryFormat = cgi.request("secondaryFormat", "");
|
||||||
if(secondaryFormat.length == 0) secondaryFormat = null;
|
if(secondaryFormat.length == 0) secondaryFormat = null;
|
||||||
|
|
||||||
JSONValue res;
|
JSONValue res;
|
||||||
|
|
||||||
// FIXME: hackalicious garbage. kill.
|
// FIXME: hackalicious garbage. kill.
|
||||||
string[][string] want = cast(string[][string]) (cgi.requestMethod == Cgi.RequestMethod.POST ? cgi.postArray : cgi.getArray);
|
want = cast(string[][string]) (cgi.requestMethod == Cgi.RequestMethod.POST ? cgi.postArray : cgi.getArray);
|
||||||
version(fb_inside_hack) {
|
version(fb_inside_hack) {
|
||||||
if(cgi.referrer.indexOf("apps.facebook.com") != -1) {
|
if(cgi.referrer.indexOf("apps.facebook.com") != -1) {
|
||||||
auto idx = cgi.referrer.indexOf("?");
|
auto idx = cgi.referrer.indexOf("?");
|
||||||
|
@ -1313,7 +1331,7 @@ void run(Provider)(Cgi cgi, Provider instantiation, size_t pathInfoStartingPoint
|
||||||
}
|
}
|
||||||
|
|
||||||
realObject.cgi = cgi;
|
realObject.cgi = cgi;
|
||||||
auto ret = fun.dispatcher(cgi, realObject, want, format, secondaryFormat);
|
ret = fun.dispatcher(cgi, realObject, want, format, secondaryFormat);
|
||||||
if(ret.completed) {
|
if(ret.completed) {
|
||||||
envelopeFormat = "no-processing";
|
envelopeFormat = "no-processing";
|
||||||
goto do_nothing_else;
|
goto do_nothing_else;
|
||||||
|
@ -1336,8 +1354,38 @@ void run(Provider)(Cgi cgi, Provider instantiation, size_t pathInfoStartingPoint
|
||||||
debug result.dFullString = e.toString();
|
debug result.dFullString = e.toString();
|
||||||
|
|
||||||
if(envelopeFormat == "document" || envelopeFormat == "html") {
|
if(envelopeFormat == "document" || envelopeFormat == "html") {
|
||||||
auto ipe = cast(InsufficientParametersException) e;
|
if(auto fve = cast(FormValidationException) e) {
|
||||||
if(ipe !is null) {
|
auto thing = fve.formFunction;
|
||||||
|
if(thing is null)
|
||||||
|
thing = fun;
|
||||||
|
fun = thing;
|
||||||
|
ret = fun.dispatcher(cgi, realObject, want, format, secondaryFormat);
|
||||||
|
result.result = ret.value;
|
||||||
|
|
||||||
|
if(fun.returnTypeIsDocument)
|
||||||
|
returnedHoldsADocument = true; // we don't replace the success flag, so this ensures no double document
|
||||||
|
|
||||||
|
moreProcessing = (Document d) {
|
||||||
|
Form f;
|
||||||
|
if(fve.getForm !is null)
|
||||||
|
f = fve.getForm(d);
|
||||||
|
else
|
||||||
|
f = d.requireSelector!Form("form");
|
||||||
|
|
||||||
|
foreach(k, v; want)
|
||||||
|
f.setValue(k, v[$-1]);
|
||||||
|
|
||||||
|
foreach(idx, failure; fve.failed) {
|
||||||
|
auto ele = f.querySelector("[name=\""~failure~"\"]");
|
||||||
|
ele.addClass("validation-failed");
|
||||||
|
ele.dataset.validationMessage = fve.messagesForUser[idx];
|
||||||
|
ele.parentNode.addChild("span", fve.messagesForUser[idx]).addClass("validation-message");
|
||||||
|
}
|
||||||
|
|
||||||
|
if(fve.postProcessor !is null)
|
||||||
|
fve.postProcessor(d, f, fve);
|
||||||
|
};
|
||||||
|
} else if(auto ipe = cast(InsufficientParametersException) e) {
|
||||||
assert(fun !is null);
|
assert(fun !is null);
|
||||||
Form form;
|
Form form;
|
||||||
if(fun.createForm !is null) {
|
if(fun.createForm !is null) {
|
||||||
|
@ -1376,6 +1424,8 @@ void run(Provider)(Cgi cgi, Provider instantiation, size_t pathInfoStartingPoint
|
||||||
|
|
||||||
foreach(k, v; cgi.get)
|
foreach(k, v; cgi.get)
|
||||||
form.setValue(k, v); // carry what we have for params over
|
form.setValue(k, v); // carry what we have for params over
|
||||||
|
foreach(k, v; cgi.post)
|
||||||
|
form.setValue(k, v); // carry what we have for params over
|
||||||
|
|
||||||
result.result.str = form.toString();
|
result.result.str = form.toString();
|
||||||
} else {
|
} else {
|
||||||
|
@ -1504,6 +1554,7 @@ void run(Provider)(Cgi cgi, Provider instantiation, size_t pathInfoStartingPoint
|
||||||
|
|
||||||
|
|
||||||
document = e.parentDocument;
|
document = e.parentDocument;
|
||||||
|
//assert(0, document.toString());
|
||||||
// FIXME: a wee bit slow, esp if func return element
|
// FIXME: a wee bit slow, esp if func return element
|
||||||
e.innerHTML = returned;
|
e.innerHTML = returned;
|
||||||
if(fun !is null)
|
if(fun !is null)
|
||||||
|
@ -1532,6 +1583,9 @@ void run(Provider)(Cgi cgi, Provider instantiation, size_t pathInfoStartingPoint
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(moreProcessing !is null)
|
||||||
|
moreProcessing(document);
|
||||||
|
|
||||||
returned = document.toString;
|
returned = document.toString;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1623,6 +1677,13 @@ mixin template FancyMain(T, Args...) {
|
||||||
/// Like FancyMain, but you can pass a custom subclass of Cgi
|
/// Like FancyMain, but you can pass a custom subclass of Cgi
|
||||||
mixin template CustomCgiFancyMain(CustomCgi, T, Args...) if(is(CustomCgi : Cgi)) {
|
mixin template CustomCgiFancyMain(CustomCgi, T, Args...) if(is(CustomCgi : Cgi)) {
|
||||||
void fancyMainFunction(Cgi cgi) { //string[] args) {
|
void fancyMainFunction(Cgi cgi) { //string[] args) {
|
||||||
|
version(catch_segfault) {
|
||||||
|
import etc.linux.memoryerror;
|
||||||
|
// NOTE: this is private on stock dmd right now, just
|
||||||
|
// open the file (src/druntime/import/etc/linux/memoryerror.d) and make it public
|
||||||
|
registerMemoryErrorHandler();
|
||||||
|
}
|
||||||
|
|
||||||
// auto cgi = new Cgi;
|
// auto cgi = new Cgi;
|
||||||
|
|
||||||
// there must be a trailing slash for relative links..
|
// there must be a trailing slash for relative links..
|
||||||
|
@ -2192,16 +2253,146 @@ Element toXmlElement(T)(Document document, T t) {
|
||||||
|
|
||||||
/// Done automatically by the wrapper function
|
/// Done automatically by the wrapper function
|
||||||
class InsufficientParametersException : Exception {
|
class InsufficientParametersException : Exception {
|
||||||
this(string functionName, string msg) {
|
this(string functionName, string msg, string file = __FILE__, size_t line = __LINE__) {
|
||||||
super(functionName ~ ": " ~ msg);
|
this.functionName = functionName;
|
||||||
|
super(functionName ~ ": " ~ msg, file, line);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
string functionName;
|
||||||
|
string argumentName;
|
||||||
|
string formLocation;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// helper for param checking
|
||||||
|
bool isValidLookingEmailAddress(string e) {
|
||||||
|
import std.net.isemail;
|
||||||
|
return isEmail(e, CheckDns.no, EmailStatusCode.any).statusCode == EmailStatusCode.valid;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Looks for things like <a or [url - the kind of stuff I often see in blatantly obvious comment spam
|
||||||
|
bool isFreeOfTypicalPlainTextSpamLinks(string txt) {
|
||||||
|
if(txt.indexOf("href=") != -1)
|
||||||
|
return false;
|
||||||
|
if(txt.indexOf("[url") != -1)
|
||||||
|
return false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
---
|
||||||
|
auto checker = new ParamCheckHelper();
|
||||||
|
|
||||||
|
checker.finish(); // this will throw if any of the checks failed
|
||||||
|
// now go ahead and use the params
|
||||||
|
---
|
||||||
|
*/
|
||||||
|
class ParamCheckHelper {
|
||||||
|
this(in Cgi cgi) {
|
||||||
|
this.cgi = cgi;
|
||||||
|
}
|
||||||
|
|
||||||
|
string[] failed;
|
||||||
|
string[] messagesForUser;
|
||||||
|
const(Cgi) cgi;
|
||||||
|
|
||||||
|
void failure(string name, string messageForUser = null) {
|
||||||
|
failed ~= name;
|
||||||
|
messagesForUser ~= messageForUser;
|
||||||
|
}
|
||||||
|
|
||||||
|
string checkParam(in string[string] among, string name, bool delegate(string) ok, string messageForUser = null) {
|
||||||
|
string value = null;
|
||||||
|
auto ptr = "name" in among;
|
||||||
|
if(ptr !is null) {
|
||||||
|
value = *ptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(!ok(value)) {
|
||||||
|
failure(name, messageForUser is null ? "Please complete this field" : messageForUser);
|
||||||
|
}
|
||||||
|
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
// int a = checkParam!int(cgi, "cool", (a) => a > 10);
|
||||||
|
T checkCgiParam(T)(string name, T defaultValue, bool delegate(T) ok, string messageForUser = null) {
|
||||||
|
auto value = cgi.request(name, defaultValue);
|
||||||
|
if(!ok(value)) {
|
||||||
|
failure(name, messageForUser);
|
||||||
|
}
|
||||||
|
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
void finish(
|
||||||
|
immutable(FunctionInfo)* formFunction,
|
||||||
|
Form delegate(Document) getForm,
|
||||||
|
void delegate(Document, Form, FormValidationException) postProcessor,
|
||||||
|
string file = __FILE__, size_t line = __LINE__)
|
||||||
|
{
|
||||||
|
if(failed.length)
|
||||||
|
throw new FormValidationException(
|
||||||
|
formFunction, getForm, postProcessor,
|
||||||
|
failed, messagesForUser,
|
||||||
|
to!string(failed), file, line);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
auto check(alias field)(ParamCheckHelper helper, bool delegate(typeof(field)) ok, string messageForUser = null) {
|
||||||
|
if(!ok(field)) {
|
||||||
|
helper.failure(field.stringof, messageForUser);
|
||||||
|
}
|
||||||
|
|
||||||
|
return field;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class FormValidationException : Exception {
|
||||||
|
this(
|
||||||
|
immutable(FunctionInfo)* formFunction,
|
||||||
|
Form delegate(Document) getForm,
|
||||||
|
void delegate(Document, Form, FormValidationException) postProcessor,
|
||||||
|
string[] failed, string[] messagesForUser,
|
||||||
|
string msg, string file = __FILE__, size_t line = __LINE__)
|
||||||
|
{
|
||||||
|
this.formFunction = formFunction;
|
||||||
|
this.getForm = getForm;
|
||||||
|
this.postProcessor = postProcessor;
|
||||||
|
this.failed = failed;
|
||||||
|
this.messagesForUser = messagesForUser;
|
||||||
|
|
||||||
|
super(msg, file, line);
|
||||||
|
}
|
||||||
|
|
||||||
|
// this will be called by the automatic catch
|
||||||
|
// it goes: Document d = formatAs(formFunction, document);
|
||||||
|
// then : Form f = getForm(d);
|
||||||
|
// it will add the values used in the current call to the form with the error conditions
|
||||||
|
// and finally, postProcessor(d, f, this);
|
||||||
|
immutable(FunctionInfo)* formFunction;
|
||||||
|
Form delegate(Document) getForm;
|
||||||
|
void delegate(Document, Form, FormValidationException) postProcessor;
|
||||||
|
string[] failed;
|
||||||
|
string[] messagesForUser;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// throw this if a paramater is invalid. Automatic forms may present this to the user in a new form. (FIXME: implement that)
|
/// throw this if a paramater is invalid. Automatic forms may present this to the user in a new form. (FIXME: implement that)
|
||||||
class InvalidParameterException : Exception {
|
class InvalidParameterException : Exception {
|
||||||
this(string param, string value, string expected) {
|
this(string param, string value, string expected, string file = __FILE__, size_t line = __LINE__) {
|
||||||
super("bad param: " ~ param ~ ". got: " ~ value ~ ". Expected: " ~expected);
|
this.param = param;
|
||||||
|
super("bad param: " ~ param ~ ". got: " ~ value ~ ". Expected: " ~expected, file, line);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
The way these are handled automatically is if something fails, web.d will
|
||||||
|
redirect the user to
|
||||||
|
|
||||||
|
formLocation ~ "?" ~ encodeVariables(cgi.get|postArray)
|
||||||
|
*/
|
||||||
|
|
||||||
|
string functionName;
|
||||||
|
string param;
|
||||||
|
string formLocation;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// convenience for throwing InvalidParameterExceptions
|
/// convenience for throwing InvalidParameterExceptions
|
||||||
|
@ -2347,7 +2538,9 @@ WrapperFunction generateWrapper(alias ObjectType, string funName, alias f, R)(Re
|
||||||
args[i] = cast() cgi.files[using]; // casting away const for the assignment to compile FIXME: shouldn't be needed
|
args[i] = cast() cgi.files[using]; // casting away const for the assignment to compile FIXME: shouldn't be needed
|
||||||
} else {
|
} else {
|
||||||
if(using !in sargs) {
|
if(using !in sargs) {
|
||||||
static if(parameterHasDefault!(f)(i)) {
|
static if(isArray!(type) && !isSomeString!(type)) {
|
||||||
|
args[i] = null;
|
||||||
|
} else static if(parameterHasDefault!(f)(i)) {
|
||||||
args[i] = mixin(parameterDefaultOf!(f)(i));
|
args[i] = mixin(parameterDefaultOf!(f)(i));
|
||||||
} else {
|
} else {
|
||||||
throw new InsufficientParametersException(funName, "arg " ~ name ~ " is not present");
|
throw new InsufficientParametersException(funName, "arg " ~ name ~ " is not present");
|
||||||
|
@ -3132,10 +3325,10 @@ struct TemplateFilters {
|
||||||
}
|
}
|
||||||
|
|
||||||
string plural(string replacement, string[] args, in Element, string) {
|
string plural(string replacement, string[] args, in Element, string) {
|
||||||
return pluralHelper(args.length ? args[0] : null, replacement);
|
return pluralHelper(args.length ? args[0] : null, replacement, args.length > 1 ? args[1] : null);
|
||||||
}
|
}
|
||||||
|
|
||||||
string pluralHelper(string number, string word) {
|
string pluralHelper(string number, string word, string pluralWord = null) {
|
||||||
if(word.length == 0)
|
if(word.length == 0)
|
||||||
return word;
|
return word;
|
||||||
|
|
||||||
|
@ -3146,12 +3339,17 @@ struct TemplateFilters {
|
||||||
if(count == 1)
|
if(count == 1)
|
||||||
return word; // it isn't actually plural
|
return word; // it isn't actually plural
|
||||||
|
|
||||||
|
if(pluralWord !is null)
|
||||||
|
return pluralWord;
|
||||||
|
|
||||||
switch(word[$ - 1]) {
|
switch(word[$ - 1]) {
|
||||||
case 's':
|
case 's':
|
||||||
case 'a', 'e', 'i', 'o', 'u':
|
case 'a', 'e', 'i', 'o', 'u':
|
||||||
return word ~ "es";
|
return word ~ "es";
|
||||||
case 'f':
|
case 'f':
|
||||||
return word[0 .. $-1] ~ "ves";
|
return word[0 .. $-1] ~ "ves";
|
||||||
|
case 'y':
|
||||||
|
return word[0 .. $-1] ~ "ies";
|
||||||
default:
|
default:
|
||||||
return word ~ "s";
|
return word ~ "s";
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue