mirror of https://github.com/adamdruppe/arsd.git
smaller buffer tbh no need for something so huge
This commit is contained in:
parent
07e19d8ff1
commit
3f15f41ddc
190
cgi.d
190
cgi.d
|
@ -6147,7 +6147,7 @@ auto formatReturnValueAsHtml(T)(T t) {
|
||||||
/++
|
/++
|
||||||
The base class for the [dispatcher] object support.
|
The base class for the [dispatcher] object support.
|
||||||
+/
|
+/
|
||||||
class WebObject() {
|
class WebObject(Helper = void) {
|
||||||
Cgi cgi;
|
Cgi cgi;
|
||||||
void initialize(Cgi cgi) {
|
void initialize(Cgi cgi) {
|
||||||
this.cgi = cgi;
|
this.cgi = cgi;
|
||||||
|
@ -6234,6 +6234,8 @@ class WebObject() {
|
||||||
FIXME: explain this better
|
FIXME: explain this better
|
||||||
+/
|
+/
|
||||||
auto serveApi(T)(string urlPrefix) {
|
auto serveApi(T)(string urlPrefix) {
|
||||||
|
assert(urlPrefix[$ - 1] == '/');
|
||||||
|
|
||||||
import arsd.dom;
|
import arsd.dom;
|
||||||
import arsd.jsvar;
|
import arsd.jsvar;
|
||||||
|
|
||||||
|
@ -6294,6 +6296,124 @@ auto serveApi(T)(string urlPrefix) {
|
||||||
return DispatcherDefinition!handler(urlPrefix, false);
|
return DispatcherDefinition!handler(urlPrefix, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/++
|
||||||
|
The base of all REST objects.
|
||||||
|
+/
|
||||||
|
class RestObject(Helper = void) {
|
||||||
|
|
||||||
|
import arsd.dom;
|
||||||
|
import arsd.jsvar;
|
||||||
|
|
||||||
|
/// Show
|
||||||
|
void show() {}
|
||||||
|
/// ditto
|
||||||
|
void show(string urlId) {
|
||||||
|
load(urlId);
|
||||||
|
show();
|
||||||
|
}
|
||||||
|
|
||||||
|
enum AccessCheck {
|
||||||
|
allowed,
|
||||||
|
denied,
|
||||||
|
nonExistant,
|
||||||
|
}
|
||||||
|
|
||||||
|
enum Operation {
|
||||||
|
show,
|
||||||
|
create,
|
||||||
|
replace,
|
||||||
|
remove,
|
||||||
|
update
|
||||||
|
}
|
||||||
|
|
||||||
|
enum UpdateResult {
|
||||||
|
accessDenied,
|
||||||
|
noSuchResource,
|
||||||
|
success,
|
||||||
|
failure,
|
||||||
|
unnecessary
|
||||||
|
}
|
||||||
|
|
||||||
|
enum ValidationResult {
|
||||||
|
valid,
|
||||||
|
invalid
|
||||||
|
}
|
||||||
|
|
||||||
|
ValidationResult delegate(typeof(this)) validateFromReflection;
|
||||||
|
Element delegate(typeof(this)) toHtmlFromReflection;
|
||||||
|
var delegate(typeof(this)) toJsonFromReflection;
|
||||||
|
|
||||||
|
/// Override this to provide access control to this object.
|
||||||
|
AccessCheck accessCheck(string urlId, Operation operation) {
|
||||||
|
return AccessCheck.allowed;
|
||||||
|
}
|
||||||
|
|
||||||
|
ValidationResult validate() {
|
||||||
|
if(validateFromReflection !is null)
|
||||||
|
return validateFromReflection(this);
|
||||||
|
return ValidationResult.valid;
|
||||||
|
}
|
||||||
|
|
||||||
|
// The functions with more arguments are the low-level ones,
|
||||||
|
// they forward to the ones with fewer arguments by default.
|
||||||
|
|
||||||
|
void create() {} // POST on a parent collection - this is called from a collection class after the members are updated
|
||||||
|
|
||||||
|
void replace() {}
|
||||||
|
void replace(string urlId, scope void delegate() applyChanges) {
|
||||||
|
load(urlId);
|
||||||
|
applyChanges();
|
||||||
|
replace();
|
||||||
|
}
|
||||||
|
|
||||||
|
void update(string[] fieldList) {}
|
||||||
|
void update(string urlId, scope void delegate() applyChanges, string[] fieldList) {
|
||||||
|
load(urlId);
|
||||||
|
applyChanges();
|
||||||
|
update(fieldList);
|
||||||
|
}
|
||||||
|
|
||||||
|
void remove() {}
|
||||||
|
|
||||||
|
void remove(string urlId) {
|
||||||
|
load(urlId);
|
||||||
|
remove();
|
||||||
|
}
|
||||||
|
|
||||||
|
abstract void load(string urlId) {}
|
||||||
|
abstract void save() {}
|
||||||
|
|
||||||
|
Element toHtml() {
|
||||||
|
if(toHtmlFromReflection)
|
||||||
|
return toHtmlFromReflection(this);
|
||||||
|
else
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
var toJson() {
|
||||||
|
if(toJsonFromReflection)
|
||||||
|
return toJsonFromReflection(this);
|
||||||
|
else
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/++
|
||||||
|
Base class for REST collections.
|
||||||
|
+/
|
||||||
|
class CollectionOf(Obj, Helper = void) : RestObject!(Helper) {
|
||||||
|
void index() {}
|
||||||
|
override void create() {}
|
||||||
|
override void load(string urlId) { assert(0); }
|
||||||
|
override void save() { assert(0); }
|
||||||
|
override void show() {
|
||||||
|
index();
|
||||||
|
}
|
||||||
|
override void show(string urlId) {
|
||||||
|
show();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/++
|
/++
|
||||||
Serves a REST object, similar to a Ruby on Rails resource.
|
Serves a REST object, similar to a Ruby on Rails resource.
|
||||||
|
|
||||||
|
@ -6375,11 +6495,17 @@ auto serveApi(T)(string urlPrefix) {
|
||||||
int id;
|
int id;
|
||||||
string name;
|
string name;
|
||||||
|
|
||||||
void show() {} // automated! GET of this specific thing
|
// the default implementations of the urlId ones is to call load(that_id) then call the arg-less one.
|
||||||
|
// but you can override them to do it differently.
|
||||||
|
|
||||||
|
// any member which is of type RestObject can be linked automatically via href btw.
|
||||||
|
|
||||||
|
void show() {}
|
||||||
|
void show(string urlId) {} // automated! GET of this specific thing
|
||||||
void create() {} // POST on a parent collection - this is called from a collection class after the members are updated
|
void create() {} // POST on a parent collection - this is called from a collection class after the members are updated
|
||||||
void replace() {} // this is the PUT; really, it just updates all fields.
|
void replace(string urlId) {} // this is the PUT; really, it just updates all fields.
|
||||||
void update() {} // PATCH, it updates some fields.
|
void update(string urlId, string[] fieldList) {} // PATCH, it updates some fields.
|
||||||
void remove() {} // DELETE
|
void remove(string urlId) {} // DELETE
|
||||||
|
|
||||||
void load(string urlId) {} // the default implementation of show() populates the id, then
|
void load(string urlId) {} // the default implementation of show() populates the id, then
|
||||||
|
|
||||||
|
@ -6405,22 +6531,52 @@ auto serveApi(T)(string urlPrefix) {
|
||||||
User create() {} // You MAY implement this, but the default is to create a new object, populate it from args, and then call create() on the child
|
User create() {} // You MAY implement this, but the default is to create a new object, populate it from args, and then call create() on the child
|
||||||
}
|
}
|
||||||
|
|
||||||
// so CollectionOf will mixin the stuff to forward it
|
|
||||||
|
|
||||||
OK, the underlying functions are actually really low level
|
|
||||||
|
|
||||||
GET(string url)
|
|
||||||
POST(string url)
|
|
||||||
PUT(string url)
|
|
||||||
|
|
||||||
The url starts with the initial thing passed.
|
|
||||||
|
|
||||||
It is the mixins that actually do the work.
|
|
||||||
+/
|
+/
|
||||||
auto serveRestObject(T)(string urlPrefix) {
|
auto serveRestObject(T)(string urlPrefix) {
|
||||||
|
assert(urlPrefix[$ - 1] != '/', "Do NOT use a trailing slash on REST objects.");
|
||||||
static bool handler(string urlPrefix, Cgi cgi) {
|
static bool handler(string urlPrefix, Cgi cgi) {
|
||||||
string url = cgi.pathInfo[urlPrefix.length .. $];
|
string url = cgi.pathInfo[urlPrefix.length .. $];
|
||||||
|
|
||||||
|
if(url.length && url[$ - 1] == '/') {
|
||||||
|
// remove the final slash...
|
||||||
|
cgi.setResponseLocation("..");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
string urlId = null;
|
||||||
|
if(url.length && url[0] == '/') {
|
||||||
|
// asking for a subobject
|
||||||
|
urlId = url[1 .. $];
|
||||||
|
foreach(idx, ch; urlId) {
|
||||||
|
if(ch == '/') {
|
||||||
|
urlId = urlId[0 .. idx];
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// FIXME: support precondition failed, if-modified-since, expectation failed, etc.
|
||||||
|
|
||||||
|
auto obj = new T();
|
||||||
|
obj.toHtmlFromReflection = delegate(t) {
|
||||||
|
import arsd.dom;
|
||||||
|
return Element.make("div", T.stringof ~ "/" ~ urlId);
|
||||||
|
};
|
||||||
|
// FIXME: populate reflection info delegates
|
||||||
|
|
||||||
|
switch(cgi.requestMethod) {
|
||||||
|
case Cgi.RequestMethod.GET:
|
||||||
|
obj.show(urlId);
|
||||||
|
cgi.write(obj.toHtml().toString, true);
|
||||||
|
break;
|
||||||
|
case Cgi.RequestMethod.POST:
|
||||||
|
case Cgi.RequestMethod.PUT:
|
||||||
|
case Cgi.RequestMethod.PATCH:
|
||||||
|
case Cgi.RequestMethod.DELETE:
|
||||||
|
default:
|
||||||
|
// FIXME: OPTIONS, HEAD
|
||||||
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
return DispatcherDefinition!handler(urlPrefix, false);
|
return DispatcherDefinition!handler(urlPrefix, false);
|
||||||
|
@ -6445,7 +6601,7 @@ auto serveStaticFile(string urlPrefix, string filename = null, string contentTyp
|
||||||
}
|
}
|
||||||
|
|
||||||
auto serveRedirect(string urlPrefix, string redirectTo) {
|
auto serveRedirect(string urlPrefix, string redirectTo) {
|
||||||
|
// FIXME
|
||||||
}
|
}
|
||||||
|
|
||||||
/+
|
/+
|
||||||
|
|
8
jpeg.d
8
jpeg.d
|
@ -3148,8 +3148,8 @@ public ubyte[] decompress_jpeg_image_from_file(bool useMalloc=false) (const(char
|
||||||
bool m_eof_flag, m_error_flag;
|
bool m_eof_flag, m_error_flag;
|
||||||
|
|
||||||
if (filename.length == 0) throw new Exception("cannot open unnamed file");
|
if (filename.length == 0) throw new Exception("cannot open unnamed file");
|
||||||
if (filename.length < 2048) {
|
if (filename.length < 512) {
|
||||||
char[2049] buffer;
|
char[513] buffer;
|
||||||
//import core.stdc.stdlib : alloca;
|
//import core.stdc.stdlib : alloca;
|
||||||
auto tfn = buffer[0 .. filename.length + 1]; // (cast(char*)alloca(filename.length+1))[0..filename.length+1];
|
auto tfn = buffer[0 .. filename.length + 1]; // (cast(char*)alloca(filename.length+1))[0..filename.length+1];
|
||||||
tfn[0..filename.length] = filename[];
|
tfn[0..filename.length] = filename[];
|
||||||
|
@ -3340,8 +3340,8 @@ public MemoryImage readJpeg (const(char)[] filename) {
|
||||||
bool m_eof_flag, m_error_flag;
|
bool m_eof_flag, m_error_flag;
|
||||||
|
|
||||||
if (filename.length == 0) throw new Exception("cannot open unnamed file");
|
if (filename.length == 0) throw new Exception("cannot open unnamed file");
|
||||||
if (filename.length < 2048) {
|
if (filename.length < 512) {
|
||||||
char[2049] buffer;
|
char[513] buffer;
|
||||||
//import core.stdc.stdlib : alloca;
|
//import core.stdc.stdlib : alloca;
|
||||||
auto tfn = buffer[0 .. filename.length + 1]; // (cast(char*)alloca(filename.length+1))[0..filename.length+1];
|
auto tfn = buffer[0 .. filename.length + 1]; // (cast(char*)alloca(filename.length+1))[0..filename.length+1];
|
||||||
tfn[0..filename.length] = filename[];
|
tfn[0..filename.length] = filename[];
|
||||||
|
|
Loading…
Reference in New Issue