This commit is contained in:
Adam D. Ruppe 2013-07-11 08:35:54 -04:00
parent c38f20fd7b
commit d614f48642
7 changed files with 349 additions and 10 deletions

3
cgi.d
View File

@ -2191,6 +2191,9 @@ string toHexUpper(long num) {
ret ~= d;
}
if(ret.length == 1)
ret ~= "0"; // url encoding requires two digits and that's what this function is used for...
return to!string(array(ret.retro));
}

View File

@ -46,7 +46,10 @@ import std.conv;
/// Like convertToUtf8, but if the encoding is unknown, it just strips all chars > 127 and calls it done instead of throwing
string convertToUtf8Lossy(immutable(ubyte)[] data, string dataCharacterEncoding) {
try {
return convertToUtf8(data, dataCharacterEncoding);
auto ret = convertToUtf8(data, dataCharacterEncoding);
import std.utf;
validate(ret);
return ret;
} catch(Exception e) {
string ret;
foreach(b; data)

8
dom.d
View File

@ -126,7 +126,7 @@ mixin template DomConvenienceFunctions() {
/// Returns whether the given class appears in this element.
bool hasClass(string c) {
auto cn = className;
string cn = className;
auto idx = cn.indexOf(c);
if(idx == -1)
@ -1268,7 +1268,7 @@ class Element {
/// This sets the style attribute with a string.
@property ElementStyle style(string s) {
this.setAttribute("style", s);
return this.style();
return this.style;
}
private void parseAttributes(string[] whichOnes = null) {
@ -1639,7 +1639,7 @@ class Element {
/// ditto
@property Element innerHTML(Html html) {
return this.innerHTML(html.source);
return this.innerHTML = html.source;
}
private void reparentTreeDocuments() {
@ -1847,7 +1847,7 @@ class Element {
Same result as innerText; the tag with all inner tags stripped out
*/
string outerText() const {
return innerText();
return innerText;
}

View File

@ -1,5 +1,25 @@
module arsd.english;
string plural(int count, string word, string pluralWord = null) {
if(count == 1 || word.length == 0)
return word; // it isn't actually plural
if(pluralWord !is null)
return pluralWord;
switch(word[$ - 1]) {
case 's':
case 'a', 'e', 'i', 'o', 'u':
return word ~ "es";
case 'f':
return word[0 .. $-1] ~ "ves";
case 'y':
return word[0 .. $-1] ~ "ies";
default:
return word ~ "s";
}
}
string numberToEnglish(long number) {
string word;
if(number == 0)

2
html.d
View File

@ -74,7 +74,7 @@ string[] htmlAttributeWhitelist = [
"href", "src", "type", "name",
"id",
"method", "enctype", // for forms only FIXME
"method", "enctype", "value", "type", // for forms only FIXME
"align", "valign", "width", "height",
];

315
mangle.d Normal file
View File

@ -0,0 +1,315 @@
module mangle;
import std.conv;
static immutable string[23] primitives = [
"char", // a
"bool", // b
"creal", // c
"double", // d
"real", // e
"float", // f
"byte", // g
"ubyte", // h
"int", // i
"ireal", // j
"uint", // k
"long", // l
"ulong", // m
null, // n
"ifloat", // o
"idouble", // p
"cfloat", // q
"cdouble", // r
"short", // s
"ushort", // t
"wchar", // u
"void", // v
"dchar", // w
];
char manglePrimitive(in char[] t) {
foreach(i, p; primitives)
if(p == t)
return cast(char) ('a' + i);
return 0;
}
import std.algorithm;
import std.array;
bool isIdentifierChar(char c) {
// FIXME: match the D spec
return c == '_' || (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || (c >= '0' && c <= '9');
}
struct StackArray(Type, size_t capacity) {
Type[capacity] buffer;
size_t length;
Type[] slice() { return buffer[0 .. length]; }
void opOpAssign(string op : "~")(Type rhs, string file = __FILE__, size_t line = __LINE__) {
if(length >= capacity) {
throw new Error("no more room", file, line);
}
buffer[length] = rhs;
length++;
}
}
char[] mangle(const(char)[] decl, char[] buffer) {
// FIXME: using this will allocate at *runtime*! Unbelievable.
// it does that even if everything is enum
auto dTokens = (sort!"a.length > b.length"(
primitives ~
[
"(",
")",
".",
",",
"!",
"[",
"]",
"*",
"const",
"immutable",
"shared",
"extern",
]));
StackArray!(const(char)[], 128) tokensBuffer;
main: while(decl.length) {
if(decl[0] == ' ' || decl[0] == '\t' || decl[0] == '\n') {
decl = decl[1 .. $];
continue;
}
foreach(token; dTokens) {
if(token is null) continue;
if(decl.length >= token.length && decl[0 .. token.length] == token) {
// make sure this isn't an identifier that coincidentally starts with a keyword
if(decl.length == token.length || !token[$ - 1].isIdentifierChar() || !decl[token.length].isIdentifierChar()) {
tokensBuffer ~= token;
decl = decl[token.length .. $];
continue main;
}
}
}
// could be an identifier or literal
int pos = 0;
while(pos < decl.length && decl[pos].isIdentifierChar)
pos++;
tokensBuffer ~= decl[0 .. pos];
decl = decl[pos .. $];
continue main;
// FIXME: literals should be handled too
}
assert(decl.length == 0); // we should have consumed all the input into tokens
auto tokens = tokensBuffer.slice();
char[64] returnTypeBuffer;
auto returnType = parseAndMangleType(tokens, returnTypeBuffer);
char[256] nameBuffer;
auto name = parseName(tokens, nameBuffer[]);
StackArray!(const(char)[], 64) arguments;
// FIXME: templates and other types of thing should be handled
assert(tokens[0] == "(", "other stuff not implemented " ~ tokens[0]);
tokens = tokens[1 .. $];
char[64][32] argTypeBuffers;
int i = 0;
while(tokens[0] != ")") {
arguments ~= parseAndMangleType(tokens, argTypeBuffers[i]);
i++;
if(tokens[0] == ",")
tokens = tokens[1 .. $];
}
assert(tokens[0] == ")", "other stuff not implemented");
return mangleFunction(name, returnType, arguments.slice(), buffer);
}
char[] parseName(ref const(char)[][] tokens, char[] nameBuffer) {
size_t where = 0;
more:
nameBuffer[where .. where + tokens[0].length] = tokens[0][];
where += tokens[0].length;
tokens = tokens[1 .. $];
if(tokens.length && tokens[0] == ".") {
tokens = tokens[1 .. $];
nameBuffer[where++] = '.';
goto more;
}
return nameBuffer[0 .. where];
}
char[] intToString(int i, char[] buffer) {
int pos = cast(int) buffer.length - 1;
if(i == 0) {
buffer[pos] = '0';
pos--;
}
while(pos > 0 && i) {
buffer[pos] = (i % 10) + '0';
pos--;
i /= 10;
}
return buffer[pos + 1 .. $];
}
char[] mangleName(in char[] name, char[] buffer) {
import std.algorithm;
import std.conv;
auto parts = name.splitter(".");
int bufferPos = 0;
foreach(part; parts) {
char[16] numberBuffer;
auto number = intToString(cast(int) part.length, numberBuffer);
buffer[bufferPos .. bufferPos + number.length] = number[];
bufferPos += number.length;
buffer[bufferPos .. bufferPos + part.length] = part[];
bufferPos += part.length;
}
return buffer[0 .. bufferPos];
}
char[] mangleFunction(in char[] name, in char[] returnTypeMangled, in char[][] argumentsMangle, char[] buffer) {
int bufferPos = 0;
buffer[bufferPos++] = '_';
buffer[bufferPos++] = 'D';
char[256] nameBuffer;
auto mn = mangleName(name, nameBuffer);
buffer[bufferPos .. bufferPos + mn.length] = mn[];
bufferPos += mn.length;
buffer[bufferPos++] = 'F';
foreach(arg; argumentsMangle) {
buffer[bufferPos .. bufferPos + arg.length] = arg[];
bufferPos += arg.length;
}
buffer[bufferPos++] = 'Z';
buffer[bufferPos .. bufferPos + returnTypeMangled.length] = returnTypeMangled[];
bufferPos += returnTypeMangled.length;
return buffer[0 .. bufferPos];
}
char[] parseAndMangleType(ref const(char)[][] tokens, char[] buffer) {
assert(tokens.length);
int bufferPos = 0;
void prepend(char p) {
for(int i = bufferPos; i > 0; i--) {
buffer[i] = buffer[i - 1];
}
buffer[0] = p;
bufferPos++;
}
// FIXME: handle all the random D type constructors
if(tokens[0] == "const" || tokens[0] == "immutable") {
if(tokens[0] == "const")
buffer[bufferPos++] = 'x';
else if(tokens[0] == "immutable")
buffer[bufferPos++] = 'y';
tokens = tokens[1 .. $];
assert(tokens[0] == "(");
tokens = tokens[1 .. $];
auto next = parseAndMangleType(tokens, buffer[bufferPos .. $]);
bufferPos += next.length;
assert(tokens[0] == ")");
tokens = tokens[1 .. $];
} else {
char primitive = manglePrimitive(tokens[0]);
if(primitive) {
buffer[bufferPos++] = primitive;
tokens = tokens[1 .. $];
} else {
// probably a struct or something, parse it as an identifier
// FIXME
char[256] nameBuffer;
auto name = parseName(tokens, nameBuffer[]);
char[256] mangledNameBuffer;
auto mn = mangleName(name, mangledNameBuffer);
buffer[bufferPos++] = 'S';
buffer[bufferPos .. bufferPos + mn.length] = mn[];
bufferPos += mn.length;
}
}
while(tokens.length) {
if(tokens[0] == "[") {
tokens = tokens[1 .. $];
prepend('A');
assert(tokens[0] == "]", "other array not implemented");
tokens = tokens[1 .. $];
} else if(tokens[0] == "*") {
prepend('P');
tokens = tokens[1 .. $];
} else break;
}
return buffer[0 .. bufferPos];
}
version(unittest) {
int foo(int, string, int);
string foo2(long, char[], int);
struct S { int a; string b; }
S foo3(S, S, string, long, int, S, int[], char[][]);
long testcomplex(int, const(const(char)[]*)[], long);
}
unittest {
import core.demangle;
char[512] buffer;
import std.stdio;
assert(mangle(demangle(foo.mangleof), buffer) == foo.mangleof);
assert(mangle(demangle(foo2.mangleof), buffer) == foo2.mangleof);
assert(mangle(demangle(foo3.mangleof), buffer) == foo3.mangleof);
assert(mangle(demangle(testcomplex.mangleof), buffer) == testcomplex.mangleof);
// FIXME: these all fail if the functions are defined inside the unittest{} block
// so still something wrong parsing those complex names or something
}
// _D6test303fooFiAyaZi
// _D6test303fooFiAyaZi
version(unittest)
void main(string[] args) {
char[512] buffer;
import std.stdio;
if(args.length > 1)
writeln(mangle(args[1], buffer));
else
writeln(mangle("int test30.foo(int, immutable(char)[])", buffer));
}

View File

@ -719,11 +719,9 @@ struct Terminal {
*/
/// Writes to the terminal at the current cursor position.
///
/// Uses std.string.xformat for the format string handling.
void writef(T...)(string f, T t) {
import std.string;
writePrintableString(xformat(f, t));
writePrintableString(format(f, t));
}
/// ditto
@ -753,7 +751,7 @@ struct Terminal {
/// Might give better performance than moveTo/writef because if the data to write matches the internal buffer, it skips sending anything (to override the buffer check, you can use moveTo and writePrintableString with ForceOption.alwaysSend)
void writefAt(T...)(int x, int y, string f, T t) {
import std.string;
auto toWrite = xformat(f, t);
auto toWrite = format(f, t);
auto oldX = _cursorX;
auto oldY = _cursorY;