mirror of https://github.com/adamdruppe/arsd.git
selector bugs
This commit is contained in:
parent
0e2017e83a
commit
c713cd6b63
1
bmp.d
1
bmp.d
|
@ -1,3 +1,4 @@
|
||||||
|
/// bmp impl for MemoryImage
|
||||||
module arsd.bmp;
|
module arsd.bmp;
|
||||||
|
|
||||||
import arsd.color;
|
import arsd.color;
|
||||||
|
|
46
cgi.d
46
cgi.d
|
@ -27,16 +27,22 @@
|
||||||
mixin GenericMain!hello;
|
mixin GenericMain!hello;
|
||||||
---
|
---
|
||||||
|
|
||||||
|
|
||||||
Compile_and_run:
|
Compile_and_run:
|
||||||
|
|
||||||
For CGI, `dmd yourfile.d cgi.d` then put the executable in your cgi-bin directory.
|
For CGI, `dmd yourfile.d cgi.d` then put the executable in your cgi-bin directory.
|
||||||
|
|
||||||
For FastCGI: `dmd yourfile.d cgi.d -version=fastcgi` and run it. spawn-fcgi helps on nginx. You can put the file in the directory for Apache. On IIS, run it with a port on the command line.
|
For FastCGI: `dmd yourfile.d cgi.d -version=fastcgi` and run it. spawn-fcgi helps on nginx. You can put the file in the directory for Apache. On IIS, run it with a port on the command line.
|
||||||
|
|
||||||
For SCGI: `dmd yourfile.d cgi.d -version=scgi` and run the executable, providing a port number on the command line.
|
For SCGI: `dmd yourfile.d cgi.d -version=scgi` and run the executable, providing a port number on the command line.
|
||||||
|
|
||||||
For an embedded HTTP server, run `dmd yourfile.d cgi.d -version=embedded_httpd` and run the generated program. It listens on port 8085 by default. You can change this on the command line with the --port option when running your program.
|
For an embedded HTTP server, run `dmd yourfile.d cgi.d -version=embedded_httpd` and run the generated program. It listens on port 8085 by default. You can change this on the command line with the --port option when running your program.
|
||||||
|
|
||||||
You can also simulate a request by passing parameters on the command line, like:
|
You can also simulate a request by passing parameters on the command line, like:
|
||||||
|
|
||||||
|
```console
|
||||||
./yourprogram GET / name=adr
|
./yourprogram GET / name=adr
|
||||||
|
```
|
||||||
|
|
||||||
And it will print the result to stdout.
|
And it will print the result to stdout.
|
||||||
|
|
||||||
|
@ -103,32 +109,43 @@
|
||||||
---
|
---
|
||||||
|
|
||||||
Concepts:
|
Concepts:
|
||||||
Input: get, post, request(), files, cookies, pathInfo, requestMethod, and HTTP headers (headers, userAgent, referrer, accept, authorization, lastEventId
|
Input: [Cgi.get], [Cgi.post], [Cgi.request], [Cgi.files], [Cgi.cookies], [Cgi.pathInfo], [Cgi.requestMethod],
|
||||||
Output: cgi.write(), cgi.header(), cgi.setResponseStatus, cgi.setResponseContentType, gzipResponse
|
and HTTP headers ([Cgi.headers], [Cgi.userAgent], [Cgi.referrer], [Cgi.accept], [Cgi.authorization], [Cgi.lastEventId]
|
||||||
Cookies: setCookie, clearCookie, cookie, cookies
|
|
||||||
Caching: cgi.setResponseExpires, cgi.updateResponseExpires, cgi.setCache
|
Output: [Cgi.write], [Cgi.header], [Cgi.setResponseStatus], [Cgi.setResponseContentType], [Cgi.gzipResponse]
|
||||||
Redirections: cgi.setResponseLocation
|
|
||||||
Other Information: remoteAddress, https, port, scriptName, requestUri, getCurrentCompleteUri, onRequestBodyDataReceived
|
Cookies: [Cgi.setCookie], [Cgi.clearCookie], [Cgi.cookie], [Cgi.cookies]
|
||||||
Overriding behavior: handleIncomingDataChunk, prepareForIncomingDataChunks, cleanUpPostDataState
|
|
||||||
|
Caching: [Cgi.setResponseExpires], [Cgi.updateResponseExpires], [Cgi.setCache]
|
||||||
|
|
||||||
|
Redirections: [Cgi.setResponseLocation]
|
||||||
|
|
||||||
|
Other Information: [Cgi.remoteAddress], [Cgi.https], [Cgi.port], [Cgi.scriptName], [Cgi.requestUri], [Cgi.getCurrentCompleteUri], [Cgi.onRequestBodyDataReceived]
|
||||||
|
|
||||||
|
Overriding behavior: [Cgi.handleIncomingDataChunk], [Cgi.prepareForIncomingDataChunks], [Cgi.cleanUpPostDataState]
|
||||||
|
|
||||||
Installing: Apache, IIS, CGI, FastCGI, SCGI, embedded HTTPD (not recommended for production use)
|
Installing: Apache, IIS, CGI, FastCGI, SCGI, embedded HTTPD (not recommended for production use)
|
||||||
|
|
||||||
Guide_for_PHP_users:
|
Guide_for_PHP_users:
|
||||||
If you are coming from PHP, here's a quick guide to help you get started:
|
If you are coming from PHP, here's a quick guide to help you get started:
|
||||||
|
|
||||||
|
```
|
||||||
$_GET["var"] == cgi.get["var"]
|
$_GET["var"] == cgi.get["var"]
|
||||||
$_POST["var"] == cgi.post["var"]
|
$_POST["var"] == cgi.post["var"]
|
||||||
$_COOKIE["var"] == cgi.cookies["var"]
|
$_COOKIE["var"] == cgi.cookies["var"]
|
||||||
|
```
|
||||||
|
|
||||||
In PHP, you can give a form element a name like "something[]", and then
|
In PHP, you can give a form element a name like `"something[]"`, and then
|
||||||
$_POST["something"] gives an array. In D, you can use whatever name
|
`$_POST["something"]` gives an array. In D, you can use whatever name
|
||||||
you want, and access an array of values with the cgi.getArray["name"] and
|
you want, and access an array of values with the `cgi.getArray["name"]` and
|
||||||
cgi.postArray["name"] members.
|
`cgi.postArray["name"]` members.
|
||||||
|
|
||||||
|
```
|
||||||
echo("hello"); == cgi.write("hello");
|
echo("hello"); == cgi.write("hello");
|
||||||
|
|
||||||
$_SERVER["REMOTE_ADDR"] == cgi.remoteAddress
|
$_SERVER["REMOTE_ADDR"] == cgi.remoteAddress
|
||||||
$_SERVER["HTTP_HOST"] == cgi.host
|
$_SERVER["HTTP_HOST"] == cgi.host
|
||||||
|
```
|
||||||
|
|
||||||
See_Also:
|
See_Also:
|
||||||
|
|
||||||
|
@ -140,9 +157,9 @@
|
||||||
|
|
||||||
Copyright:
|
Copyright:
|
||||||
|
|
||||||
cgi.d copyright 2008-2015, Adam D. Ruppe. Provided under the Boost Software License.
|
cgi.d copyright 2008-2016, Adam D. Ruppe. Provided under the Boost Software License.
|
||||||
|
|
||||||
Yes, this file is seven years old, and yes, it is still actively maintained and used.
|
Yes, this file is almost eight years old, and yes, it is still actively maintained and used.
|
||||||
+/
|
+/
|
||||||
module arsd.cgi;
|
module arsd.cgi;
|
||||||
|
|
||||||
|
@ -3107,6 +3124,7 @@ import std.socket;
|
||||||
|
|
||||||
// it is a class primarily for reference semantics
|
// it is a class primarily for reference semantics
|
||||||
// I might change this interface
|
// I might change this interface
|
||||||
|
///
|
||||||
class BufferedInputRange {
|
class BufferedInputRange {
|
||||||
version(Posix)
|
version(Posix)
|
||||||
this(int source, ubyte[] buffer = null) {
|
this(int source, ubyte[] buffer = null) {
|
||||||
|
@ -3644,6 +3662,7 @@ version(cgi_with_websocket) {
|
||||||
WEBSOCKET SUPPORT:
|
WEBSOCKET SUPPORT:
|
||||||
|
|
||||||
Full example:
|
Full example:
|
||||||
|
---
|
||||||
import arsd.cgi;
|
import arsd.cgi;
|
||||||
|
|
||||||
void websocketEcho(Cgi cgi) {
|
void websocketEcho(Cgi cgi) {
|
||||||
|
@ -3671,6 +3690,7 @@ version(cgi_with_websocket) {
|
||||||
}
|
}
|
||||||
|
|
||||||
mixin GenericMain!websocketEcho;
|
mixin GenericMain!websocketEcho;
|
||||||
|
---
|
||||||
*/
|
*/
|
||||||
|
|
||||||
class WebSocket {
|
class WebSocket {
|
||||||
|
|
6
cidr.d
6
cidr.d
|
@ -1,3 +1,7 @@
|
||||||
|
///
|
||||||
|
module cidr;
|
||||||
|
|
||||||
|
///
|
||||||
uint addressToUint(string address) {
|
uint addressToUint(string address) {
|
||||||
import std.algorithm.iteration, std.conv;
|
import std.algorithm.iteration, std.conv;
|
||||||
|
|
||||||
|
@ -12,6 +16,7 @@ uint addressToUint(string address) {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///
|
||||||
string uintToAddress(uint addr) {
|
string uintToAddress(uint addr) {
|
||||||
import std.conv;
|
import std.conv;
|
||||||
string res;
|
string res;
|
||||||
|
@ -26,6 +31,7 @@ string uintToAddress(uint addr) {
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///
|
||||||
struct IPv4Block {
|
struct IPv4Block {
|
||||||
this(string cidr) {
|
this(string cidr) {
|
||||||
import std.algorithm.searching, std.conv;
|
import std.algorithm.searching, std.conv;
|
||||||
|
|
33
color.d
33
color.d
|
@ -1,3 +1,4 @@
|
||||||
|
///
|
||||||
module arsd.color;
|
module arsd.color;
|
||||||
|
|
||||||
@safe:
|
@safe:
|
||||||
|
@ -114,7 +115,7 @@ private {
|
||||||
/// Represents an RGBA color
|
/// Represents an RGBA color
|
||||||
struct Color {
|
struct Color {
|
||||||
union {
|
union {
|
||||||
ubyte[4] components;
|
ubyte[4] components; ///
|
||||||
|
|
||||||
struct {
|
struct {
|
||||||
ubyte r; /// red
|
ubyte r; /// red
|
||||||
|
@ -123,7 +124,7 @@ struct Color {
|
||||||
ubyte a; /// alpha. 255 == opaque
|
ubyte a; /// alpha. 255 == opaque
|
||||||
}
|
}
|
||||||
|
|
||||||
uint asUint;
|
uint asUint; ///
|
||||||
}
|
}
|
||||||
|
|
||||||
// this makes sure they are in range before casting
|
// this makes sure they are in range before casting
|
||||||
|
@ -518,25 +519,28 @@ Color setLightness(Color c, real lightness) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
///
|
||||||
Color rotateHue(Color c, real degrees) {
|
Color rotateHue(Color c, real degrees) {
|
||||||
auto hsl = toHsl(c);
|
auto hsl = toHsl(c);
|
||||||
hsl[0] += degrees;
|
hsl[0] += degrees;
|
||||||
return fromHsl(hsl);
|
return fromHsl(hsl);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///
|
||||||
Color setHue(Color c, real hue) {
|
Color setHue(Color c, real hue) {
|
||||||
auto hsl = toHsl(c);
|
auto hsl = toHsl(c);
|
||||||
hsl[0] = hue;
|
hsl[0] = hue;
|
||||||
return fromHsl(hsl);
|
return fromHsl(hsl);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///
|
||||||
Color desaturate(Color c, real percentage) {
|
Color desaturate(Color c, real percentage) {
|
||||||
auto hsl = toHsl(c);
|
auto hsl = toHsl(c);
|
||||||
hsl[1] *= (1 - percentage);
|
hsl[1] *= (1 - percentage);
|
||||||
return fromHsl(hsl);
|
return fromHsl(hsl);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///
|
||||||
Color saturate(Color c, real percentage) {
|
Color saturate(Color c, real percentage) {
|
||||||
auto hsl = toHsl(c);
|
auto hsl = toHsl(c);
|
||||||
hsl[1] *= (1 + percentage);
|
hsl[1] *= (1 + percentage);
|
||||||
|
@ -545,6 +549,7 @@ Color saturate(Color c, real percentage) {
|
||||||
return fromHsl(hsl);
|
return fromHsl(hsl);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///
|
||||||
Color setSaturation(Color c, real saturation) {
|
Color setSaturation(Color c, real saturation) {
|
||||||
auto hsl = toHsl(c);
|
auto hsl = toHsl(c);
|
||||||
hsl[1] = saturation;
|
hsl[1] = saturation;
|
||||||
|
@ -592,6 +597,7 @@ So, given the background color and the resultant color, what was
|
||||||
composited on to it?
|
composited on to it?
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
///
|
||||||
ubyte unalpha(ubyte colorYouHave, float alpha, ubyte backgroundColor) {
|
ubyte unalpha(ubyte colorYouHave, float alpha, ubyte backgroundColor) {
|
||||||
// resultingColor = (1-alpha) * backgroundColor + alpha * answer
|
// resultingColor = (1-alpha) * backgroundColor + alpha * answer
|
||||||
auto resultingColorf = cast(float) colorYouHave;
|
auto resultingColorf = cast(float) colorYouHave;
|
||||||
|
@ -605,6 +611,7 @@ ubyte unalpha(ubyte colorYouHave, float alpha, ubyte backgroundColor) {
|
||||||
return cast(ubyte) answer;
|
return cast(ubyte) answer;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///
|
||||||
ubyte makeAlpha(ubyte colorYouHave, ubyte backgroundColor/*, ubyte foreground = 0x00*/) {
|
ubyte makeAlpha(ubyte colorYouHave, ubyte backgroundColor/*, ubyte foreground = 0x00*/) {
|
||||||
//auto foregroundf = cast(float) foreground;
|
//auto foregroundf = cast(float) foreground;
|
||||||
auto foregroundf = 0.00f;
|
auto foregroundf = 0.00f;
|
||||||
|
@ -644,6 +651,7 @@ int fromHex(string s) {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///
|
||||||
Color colorFromString(string s) {
|
Color colorFromString(string s) {
|
||||||
if(s.length == 0)
|
if(s.length == 0)
|
||||||
return Color(0,0,0,255);
|
return Color(0,0,0,255);
|
||||||
|
@ -1118,19 +1126,22 @@ void floydSteinbergDither(IndexedImage img, in TrueColorImage original) {
|
||||||
|
|
||||||
// these are just really useful in a lot of places where the color/image functions are used,
|
// these are just really useful in a lot of places where the color/image functions are used,
|
||||||
// so I want them available with Color
|
// so I want them available with Color
|
||||||
|
///
|
||||||
struct Point {
|
struct Point {
|
||||||
int x;
|
int x; ///
|
||||||
int y;
|
int y; ///
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///
|
||||||
struct Size {
|
struct Size {
|
||||||
int width;
|
int width; ///
|
||||||
int height;
|
int height; ///
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///
|
||||||
struct Rectangle {
|
struct Rectangle {
|
||||||
int left;
|
int left; ///
|
||||||
int top;
|
int top; ///
|
||||||
int right;
|
int right; ///
|
||||||
int bottom;
|
int bottom; ///
|
||||||
}
|
}
|
||||||
|
|
2
csv.d
2
csv.d
|
@ -1,8 +1,10 @@
|
||||||
|
///
|
||||||
module arsd.csv;
|
module arsd.csv;
|
||||||
|
|
||||||
import std.string;
|
import std.string;
|
||||||
import std.array;
|
import std.array;
|
||||||
|
|
||||||
|
///
|
||||||
string[][] readCsv(string data) {
|
string[][] readCsv(string data) {
|
||||||
data = data.replace("\r", "");
|
data = data.replace("\r", "");
|
||||||
|
|
||||||
|
|
1
curl.d
1
curl.d
|
@ -1,3 +1,4 @@
|
||||||
|
/// curl wrapper, it sux
|
||||||
module arsd.curl;
|
module arsd.curl;
|
||||||
|
|
||||||
// see this for info on making a curl.lib on windows:
|
// see this for info on making a curl.lib on windows:
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
///
|
||||||
module arsd.database;
|
module arsd.database;
|
||||||
|
|
||||||
public import std.variant;
|
public import std.variant;
|
||||||
|
@ -21,6 +22,7 @@ import std.string;
|
||||||
so they are always done once per program...
|
so they are always done once per program...
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
///
|
||||||
interface Database {
|
interface Database {
|
||||||
/// Actually implements the query for the database. The query() method
|
/// Actually implements the query for the database. The query() method
|
||||||
/// below might be easier to use.
|
/// below might be easier to use.
|
||||||
|
|
46
dom.d
46
dom.d
|
@ -5051,11 +5051,9 @@ int intFromHex(string hex) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
foreach(a; notSelectors) {
|
foreach(a; notSelectors) {
|
||||||
auto sels = parseSelectorString(a);
|
auto sel = Selector(a);
|
||||||
foreach(sel; sels)
|
if(sel.matchesElement(e))
|
||||||
foreach(part; sel.parts)
|
return false;
|
||||||
if(part.matchElement(e))
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
@ -5092,23 +5090,9 @@ int intFromHex(string hex) {
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case 2: // next-sibling
|
case 2: // next-sibling
|
||||||
auto tmp = start.parentNode;
|
auto e = start.nextSibling("*");
|
||||||
if(tmp !is null) {
|
if(part.matchElement(e))
|
||||||
sizediff_t pos = -1;
|
ret ~= getElementsBySelectorParts(e, parts[1..$]);
|
||||||
auto children = tmp.childElements;
|
|
||||||
foreach(i, child; children) {
|
|
||||||
if(child is start) {
|
|
||||||
pos = i;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
assert(pos != -1);
|
|
||||||
if(pos + 1 < children.length) {
|
|
||||||
auto e = children[pos+1];
|
|
||||||
if(part.matchElement(e))
|
|
||||||
ret ~= getElementsBySelectorParts(e, parts[1..$]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case 3: // younger sibling
|
case 3: // younger sibling
|
||||||
auto tmp = start.parentNode;
|
auto tmp = start.parentNode;
|
||||||
|
@ -5204,6 +5188,16 @@ int intFromHex(string hex) {
|
||||||
return removeDuplicates(ret);
|
return removeDuplicates(ret);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/++
|
||||||
|
Like [getMatchingElements], but returns a lazy range. Be careful
|
||||||
|
about mutating the dom as you iterate through this.
|
||||||
|
+/
|
||||||
|
auto getMatchingElementsLazy(Element start, Element relativeTo = null) {
|
||||||
|
import std.algorithm.iteration;
|
||||||
|
return start.tree.filter!(a => this.matchesElement(a, relativeTo));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/// Returns the string this was built from
|
/// Returns the string this was built from
|
||||||
string toString() {
|
string toString() {
|
||||||
return original;
|
return original;
|
||||||
|
@ -5254,7 +5248,7 @@ int intFromHex(string hex) {
|
||||||
int lastSeparation = -1;
|
int lastSeparation = -1;
|
||||||
foreach(part; retro(parts)) {
|
foreach(part; retro(parts)) {
|
||||||
|
|
||||||
writeln("matching ", where, " with ", part, " via ", lastSeparation);
|
// writeln("matching ", where, " with ", part, " via ", lastSeparation);
|
||||||
|
|
||||||
if(lastSeparation == -1) {
|
if(lastSeparation == -1) {
|
||||||
if(!part.matchElement(where))
|
if(!part.matchElement(where))
|
||||||
|
@ -5280,12 +5274,12 @@ int intFromHex(string hex) {
|
||||||
if(!part.matchElement(where))
|
if(!part.matchElement(where))
|
||||||
return false;
|
return false;
|
||||||
} else if(lastSeparation == 2) { // the + operator
|
} else if(lastSeparation == 2) { // the + operator
|
||||||
where = where.previousSibling;
|
where = where.previousSibling("*");
|
||||||
|
|
||||||
if(!part.matchElement(where))
|
if(!part.matchElement(where))
|
||||||
return false;
|
return false;
|
||||||
} else if(lastSeparation == 3) { // the ~ operator
|
} else if(lastSeparation == 3) { // the ~ operator
|
||||||
where = where.previousSibling;
|
where = where.previousSibling("*");
|
||||||
while(where !is null) {
|
while(where !is null) {
|
||||||
if(part.matchElement(where))
|
if(part.matchElement(where))
|
||||||
break;
|
break;
|
||||||
|
@ -5293,7 +5287,7 @@ int intFromHex(string hex) {
|
||||||
if(where is relativeTo)
|
if(where is relativeTo)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
where = where.previousSibling;
|
where = where.previousSibling("*");
|
||||||
}
|
}
|
||||||
|
|
||||||
if(where is null)
|
if(where is null)
|
||||||
|
|
1
email.d
1
email.d
|
@ -1,3 +1,4 @@
|
||||||
|
///
|
||||||
module arsd.email;
|
module arsd.email;
|
||||||
|
|
||||||
import std.net.curl;
|
import std.net.curl;
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
///
|
||||||
module arsd.english;
|
module arsd.english;
|
||||||
|
|
||||||
string plural(int count, string word, string pluralWord = null) {
|
string plural(int count, string word, string pluralWord = null) {
|
||||||
|
|
22
eventloop.d
22
eventloop.d
|
@ -1,3 +1,4 @@
|
||||||
|
/// crappy event loop for linux
|
||||||
module arsd.eventloop;
|
module arsd.eventloop;
|
||||||
|
|
||||||
/* **** */
|
/* **** */
|
||||||
|
@ -251,6 +252,7 @@ public struct FileEventDispatcher {
|
||||||
return;
|
return;
|
||||||
|
|
||||||
addListener(&lowLevelReadHandler);
|
addListener(&lowLevelReadHandler);
|
||||||
|
addListener(&lowLevelHupHandler);
|
||||||
addListener(&lowLevelWriteHandler);
|
addListener(&lowLevelWriteHandler);
|
||||||
addListener(&lowLevelErrorHandler);
|
addListener(&lowLevelErrorHandler);
|
||||||
handlersActive = true;
|
handlersActive = true;
|
||||||
|
@ -261,6 +263,7 @@ public struct FileEventDispatcher {
|
||||||
return;
|
return;
|
||||||
|
|
||||||
removeListener(&lowLevelErrorHandler);
|
removeListener(&lowLevelErrorHandler);
|
||||||
|
removeListener(&lowLevelHupHandler);
|
||||||
removeListener(&lowLevelWriteHandler);
|
removeListener(&lowLevelWriteHandler);
|
||||||
removeListener(&lowLevelReadHandler);
|
removeListener(&lowLevelReadHandler);
|
||||||
handlersActive = false;
|
handlersActive = false;
|
||||||
|
@ -302,6 +305,9 @@ public struct FileEventDispatcher {
|
||||||
private void lowLevelErrorHandler(FileError ev) {
|
private void lowLevelErrorHandler(FileError ev) {
|
||||||
doHandler(ev.fd, 2);
|
doHandler(ev.fd, 2);
|
||||||
}
|
}
|
||||||
|
private void lowLevelHupHandler(FileHup ev) {
|
||||||
|
doHandler(ev.fd, 2);
|
||||||
|
}
|
||||||
|
|
||||||
/// You can add a file to listen to here. Files can be OS handles or Phobos types. The handlers can be null, meaning use the default
|
/// You can add a file to listen to here. Files can be OS handles or Phobos types. The handlers can be null, meaning use the default
|
||||||
/// (see: setDefaultHandler), or callables with zero or one argument. If they take an argument, it will be the file being handled at this time.
|
/// (see: setDefaultHandler), or callables with zero or one argument. If they take an argument, it will be the file being handled at this time.
|
||||||
|
@ -330,11 +336,11 @@ public struct FileEventDispatcher {
|
||||||
events |= FileEvents.read;
|
events |= FileEvents.read;
|
||||||
}
|
}
|
||||||
if(writeEventHandler !is null) {
|
if(writeEventHandler !is null) {
|
||||||
handlerSet[0] = wrap(writeEventHandler);
|
handlerSet[1] = wrap(writeEventHandler);
|
||||||
events |= FileEvents.write;
|
events |= FileEvents.write;
|
||||||
}
|
}
|
||||||
if(errorEventHandler !is null)
|
if(errorEventHandler !is null)
|
||||||
handlerSet[0] = wrap(errorEventHandler);
|
handlerSet[2] = wrap(errorEventHandler);
|
||||||
|
|
||||||
listeners[handle] = handlerSet;
|
listeners[handle] = handlerSet;
|
||||||
|
|
||||||
|
@ -344,6 +350,7 @@ public struct FileEventDispatcher {
|
||||||
|
|
||||||
public void removeFile(OsFileHandle handle) {
|
public void removeFile(OsFileHandle handle) {
|
||||||
listeners.remove(handle);
|
listeners.remove(handle);
|
||||||
|
removeFileFromLoopImplementation(handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// What should this default handler work on?
|
/// What should this default handler work on?
|
||||||
|
@ -492,6 +499,12 @@ version(linux) {
|
||||||
epoll_ctl(epoll, EPOLL_CTL_ADD, fd, &ev);
|
epoll_ctl(epoll, EPOLL_CTL_ADD, fd, &ev);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void removeFileFromLoopImplementation(int fd) {
|
||||||
|
epoll_event ev = void;
|
||||||
|
ev.data.fd = fd;
|
||||||
|
epoll_ctl(epoll, EPOLL_CTL_DEL, fd, &ev);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
private void loopImplementation() {
|
private void loopImplementation() {
|
||||||
insideLoop = true;
|
insideLoop = true;
|
||||||
|
@ -561,7 +574,10 @@ version(linux) {
|
||||||
if((flags & EPOLLERR)) {
|
if((flags & EPOLLERR)) {
|
||||||
//import core.stdc.stdio; printf("ERROR on fd from epoll %d\n", fd);
|
//import core.stdc.stdio; printf("ERROR on fd from epoll %d\n", fd);
|
||||||
sendSync(FileError(fd));
|
sendSync(FileError(fd));
|
||||||
break outer_loop;
|
|
||||||
|
// I automatically remove them because otherwise the error flag
|
||||||
|
// may never actually be cleared and this thing will infinite loop.
|
||||||
|
removeFileEventListeners(fd);
|
||||||
}
|
}
|
||||||
if((flags & EPOLLHUP)) {
|
if((flags & EPOLLHUP)) {
|
||||||
//import core.stdc.stdio; printf("HUP on fd from epoll %d\n", fd);
|
//import core.stdc.stdio; printf("HUP on fd from epoll %d\n", fd);
|
||||||
|
|
|
@ -1,3 +1,5 @@
|
||||||
|
/// A draft of a better way to do exceptions
|
||||||
|
module arsd.exception;
|
||||||
/*
|
/*
|
||||||
Exceptions 2.0
|
Exceptions 2.0
|
||||||
*/
|
*/
|
||||||
|
|
1
hmac.d
1
hmac.d
|
@ -1,3 +1,4 @@
|
||||||
|
///
|
||||||
module arsd.hmac;
|
module arsd.hmac;
|
||||||
|
|
||||||
// FIXME: the blocksize is correct for MD5, SHA1, and SHA256 but not generally
|
// FIXME: the blocksize is correct for MD5, SHA1, and SHA256 but not generally
|
||||||
|
|
127
html.d
127
html.d
|
@ -1,5 +1,5 @@
|
||||||
/**
|
/**
|
||||||
This module includes functions to work with HTML.
|
This module includes functions to work with HTML and CSS.
|
||||||
|
|
||||||
It publically imports the DOM module to get started.
|
It publically imports the DOM module to get started.
|
||||||
Then it adds a number of functions to enhance html
|
Then it adds a number of functions to enhance html
|
||||||
|
@ -1235,6 +1235,7 @@ string translateJavascriptSourceWithDToStandardScript(string src)() {
|
||||||
+/
|
+/
|
||||||
|
|
||||||
abstract class CssPart {
|
abstract class CssPart {
|
||||||
|
string comment;
|
||||||
override string toString() const;
|
override string toString() const;
|
||||||
CssPart clone() const;
|
CssPart clone() const;
|
||||||
}
|
}
|
||||||
|
@ -1245,6 +1246,7 @@ class CssAtRule : CssPart {
|
||||||
assert(css.length);
|
assert(css.length);
|
||||||
assert(css[0] == '@');
|
assert(css[0] == '@');
|
||||||
|
|
||||||
|
auto cssl = css.length;
|
||||||
int braceCount = 0;
|
int braceCount = 0;
|
||||||
int startOfInnerSlice = -1;
|
int startOfInnerSlice = -1;
|
||||||
|
|
||||||
|
@ -1277,6 +1279,12 @@ class CssAtRule : CssPart {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(cssl == css.length) {
|
||||||
|
throw new Exception("Bad CSS: unclosed @ rule. " ~ to!string(braceCount) ~ " brace(s) uncloced");
|
||||||
|
}
|
||||||
|
|
||||||
|
innerParts = lexCss(inner, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
string content;
|
string content;
|
||||||
|
@ -1284,17 +1292,35 @@ class CssAtRule : CssPart {
|
||||||
string opener;
|
string opener;
|
||||||
string inner;
|
string inner;
|
||||||
|
|
||||||
|
CssPart[] innerParts;
|
||||||
|
|
||||||
override CssAtRule clone() const {
|
override CssAtRule clone() const {
|
||||||
auto n = new CssAtRule();
|
auto n = new CssAtRule();
|
||||||
n.content = content;
|
n.content = content;
|
||||||
n.opener = opener;
|
n.opener = opener;
|
||||||
n.inner = inner;
|
n.inner = inner;
|
||||||
|
foreach(part; innerParts)
|
||||||
|
n.innerParts ~= part.clone();
|
||||||
return n;
|
return n;
|
||||||
}
|
}
|
||||||
override string toString() const { return content; }
|
override string toString() const {
|
||||||
}
|
string c;
|
||||||
|
if(comment.length)
|
||||||
|
c ~= "/* " ~ comment ~ "*/\n";
|
||||||
|
c ~= opener.strip();
|
||||||
|
if(innerParts.length) {
|
||||||
|
string i;
|
||||||
|
foreach(part; innerParts)
|
||||||
|
i ~= part.toString() ~ "\n";
|
||||||
|
|
||||||
import std.stdio;
|
c ~= " {\n";
|
||||||
|
foreach(line; i.splitLines)
|
||||||
|
c ~= "\t" ~ line ~ "\n";
|
||||||
|
c ~= "}";
|
||||||
|
}
|
||||||
|
return c;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
class CssRuleSet : CssPart {
|
class CssRuleSet : CssPart {
|
||||||
this() {}
|
this() {}
|
||||||
|
@ -1326,7 +1352,7 @@ class CssRuleSet : CssPart {
|
||||||
f++;
|
f++;
|
||||||
css = css[f .. $];
|
css = css[f .. $];
|
||||||
|
|
||||||
contents = lexCss(content);
|
contents = lexCss(content, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
string[] selectors;
|
string[] selectors;
|
||||||
|
@ -1418,6 +1444,10 @@ class CssRuleSet : CssPart {
|
||||||
override string toString() const {
|
override string toString() const {
|
||||||
string ret;
|
string ret;
|
||||||
|
|
||||||
|
|
||||||
|
if(comment.length)
|
||||||
|
ret ~= "/* " ~ comment ~ "*/\n";
|
||||||
|
|
||||||
bool outputtedSelector = false;
|
bool outputtedSelector = false;
|
||||||
foreach(selector; selectors) {
|
foreach(selector; selectors) {
|
||||||
if(outputtedSelector)
|
if(outputtedSelector)
|
||||||
|
@ -1456,6 +1486,21 @@ class CssRule : CssPart {
|
||||||
// note: does not include the ending semicolon
|
// note: does not include the ending semicolon
|
||||||
string content;
|
string content;
|
||||||
|
|
||||||
|
string key() const {
|
||||||
|
auto idx = content.indexOf(":");
|
||||||
|
if(idx == -1)
|
||||||
|
throw new Exception("Bad css, missing colon in " ~ content);
|
||||||
|
return content[0 .. idx].strip.toLower;
|
||||||
|
}
|
||||||
|
|
||||||
|
string value() const {
|
||||||
|
auto idx = content.indexOf(":");
|
||||||
|
if(idx == -1)
|
||||||
|
throw new Exception("Bad css, missing colon in " ~ content);
|
||||||
|
|
||||||
|
return content[idx + 1 .. $].strip;
|
||||||
|
}
|
||||||
|
|
||||||
override CssRule clone() const {
|
override CssRule clone() const {
|
||||||
auto n = new CssRule();
|
auto n = new CssRule();
|
||||||
n.content = content;
|
n.content = content;
|
||||||
|
@ -1463,20 +1508,32 @@ class CssRule : CssPart {
|
||||||
}
|
}
|
||||||
|
|
||||||
override string toString() const {
|
override string toString() const {
|
||||||
|
string ret;
|
||||||
if(strip(content).length == 0)
|
if(strip(content).length == 0)
|
||||||
return "";
|
ret = "";
|
||||||
return content ~ ";";
|
else
|
||||||
|
ret = key ~ ": " ~ value ~ ";";
|
||||||
|
|
||||||
|
if(comment.length)
|
||||||
|
ret ~= " /* " ~ comment ~ " */";
|
||||||
|
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
CssPart[] lexCss(string css) {
|
// Never call stripComments = false unless you have already stripped them.
|
||||||
import std.regex;
|
// this thing can't actually handle comments intelligently.
|
||||||
// strips comments
|
CssPart[] lexCss(string css, bool stripComments = true) {
|
||||||
css = std.regex.replace(css, regex(r"\/\*[^*]*\*+([^/*][^*]*\*+)*\/", "g"), "");
|
if(stripComments) {
|
||||||
|
import std.regex;
|
||||||
|
css = std.regex.replace(css, regex(r"\/\*[^*]*\*+([^/*][^*]*\*+)*\/", "g"), "");
|
||||||
|
}
|
||||||
|
|
||||||
CssPart[] ret;
|
CssPart[] ret;
|
||||||
css = css.stripLeft();
|
css = css.stripLeft();
|
||||||
|
|
||||||
|
int cnt;
|
||||||
|
|
||||||
while(css.length > 1) {
|
while(css.length > 1) {
|
||||||
CssPart p;
|
CssPart p;
|
||||||
|
|
||||||
|
@ -1486,7 +1543,7 @@ CssPart[] lexCss(string css) {
|
||||||
// non-at rules can be either rules or sets.
|
// non-at rules can be either rules or sets.
|
||||||
// The question is: which comes first, the ';' or the '{' ?
|
// The question is: which comes first, the ';' or the '{' ?
|
||||||
|
|
||||||
auto endOfStatement = css.indexOf(";");
|
auto endOfStatement = css.indexOfCssSmart(';');
|
||||||
if(endOfStatement == -1)
|
if(endOfStatement == -1)
|
||||||
endOfStatement = css.indexOf("}");
|
endOfStatement = css.indexOf("}");
|
||||||
if(endOfStatement == -1)
|
if(endOfStatement == -1)
|
||||||
|
@ -1508,6 +1565,46 @@ CssPart[] lexCss(string css) {
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// This needs to skip characters inside parens or quotes, so it
|
||||||
|
// doesn't trip up on stuff like data uris when looking for a terminating
|
||||||
|
// character.
|
||||||
|
ptrdiff_t indexOfCssSmart(string i, char find) {
|
||||||
|
int parenCount;
|
||||||
|
char quote;
|
||||||
|
bool escaping;
|
||||||
|
foreach(idx, ch; i) {
|
||||||
|
if(escaping) {
|
||||||
|
escaping = false;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if(quote != char.init) {
|
||||||
|
if(ch == quote)
|
||||||
|
quote = char.init;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if(ch == '\'' || ch == '"') {
|
||||||
|
quote = ch;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(ch == '(')
|
||||||
|
parenCount++;
|
||||||
|
|
||||||
|
if(parenCount) {
|
||||||
|
if(ch == ')')
|
||||||
|
parenCount--;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// at this point, we are not in parenthesis nor are we in
|
||||||
|
// a quote, so we can actually search for the relevant character
|
||||||
|
|
||||||
|
if(ch == find)
|
||||||
|
return idx;
|
||||||
|
}
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
string cssToString(in CssPart[] css) {
|
string cssToString(in CssPart[] css) {
|
||||||
string ret;
|
string ret;
|
||||||
foreach(c; css) {
|
foreach(c; css) {
|
||||||
|
@ -1543,7 +1640,7 @@ const(CssPart)[] denestCss(CssPart[] css) {
|
||||||
auto newCss = at.opener ~ "{\n";
|
auto newCss = at.opener ~ "{\n";
|
||||||
|
|
||||||
// the whitespace manipulations are just a crude indentation thing
|
// the whitespace manipulations are just a crude indentation thing
|
||||||
newCss ~= "\t" ~ (cssToString(denestCss(lexCss(at.inner))).replace("\n", "\n\t").replace("\n\t\n\t", "\n\n\t"));
|
newCss ~= "\t" ~ (cssToString(denestCss(lexCss(at.inner, false))).replace("\n", "\n\t").replace("\n\t\n\t", "\n\n\t"));
|
||||||
|
|
||||||
newCss ~= "\n}";
|
newCss ~= "\n}";
|
||||||
|
|
||||||
|
@ -2161,11 +2258,11 @@ Color readCssColor(string cssColor) {
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Copyright: Adam D. Ruppe, 2010 - 2012
|
Copyright: Adam D. Ruppe, 2010 - 2015
|
||||||
License: <a href="http://www.boost.org/LICENSE_1_0.txt">Boost License 1.0</a>.
|
License: <a href="http://www.boost.org/LICENSE_1_0.txt">Boost License 1.0</a>.
|
||||||
Authors: Adam D. Ruppe, with contributions by Nick Sabalausky and Trass3r
|
Authors: Adam D. Ruppe, with contributions by Nick Sabalausky and Trass3r
|
||||||
|
|
||||||
Copyright Adam D. Ruppe 2010-2012.
|
Copyright Adam D. Ruppe 2010-2015.
|
||||||
Distributed under the Boost Software License, Version 1.0.
|
Distributed under the Boost Software License, Version 1.0.
|
||||||
(See accompanying file LICENSE_1_0.txt or copy at
|
(See accompanying file LICENSE_1_0.txt or copy at
|
||||||
http://www.boost.org/LICENSE_1_0.txt)
|
http://www.boost.org/LICENSE_1_0.txt)
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
///
|
||||||
module arsd.htmltotext;
|
module arsd.htmltotext;
|
||||||
|
|
||||||
import arsd.dom;
|
import arsd.dom;
|
||||||
|
|
1
http2.d
1
http2.d
|
@ -1,3 +1,4 @@
|
||||||
|
/// HTTP client lib
|
||||||
// Copyright 2013, Adam D. Ruppe.
|
// Copyright 2013, Adam D. Ruppe.
|
||||||
module arsd.http2;
|
module arsd.http2;
|
||||||
|
|
||||||
|
|
3
jpg.d
3
jpg.d
|
@ -1,3 +1,4 @@
|
||||||
|
///
|
||||||
module arsd.jpg;
|
module arsd.jpg;
|
||||||
|
|
||||||
import std.typecons;
|
import std.typecons;
|
||||||
|
@ -88,6 +89,7 @@ Tuple!(int, int) getSizeFromFile(string filename) {
|
||||||
}
|
}
|
||||||
|
|
||||||
version(with_libjpeg) {
|
version(with_libjpeg) {
|
||||||
|
/+
|
||||||
import arsd.color;
|
import arsd.color;
|
||||||
|
|
||||||
TrueColorImage read_JPEG_file(string filename) {
|
TrueColorImage read_JPEG_file(string filename) {
|
||||||
|
@ -213,4 +215,5 @@ version(with_libjpeg) {
|
||||||
/* And we're done! */
|
/* And we're done! */
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
+/
|
||||||
}
|
}
|
||||||
|
|
11
libssh2.d
11
libssh2.d
|
@ -1,6 +1,6 @@
|
||||||
// Small wrapper for libssh2
|
/// Small wrapper for libssh2
|
||||||
// just link with it on Linux
|
/// just link with it on Linux
|
||||||
// it'll need a couple dlls and a lib on windows.
|
/// it'll need a couple dlls and a lib on windows.
|
||||||
|
|
||||||
module arsd.libssh2;
|
module arsd.libssh2;
|
||||||
|
|
||||||
|
@ -181,4 +181,9 @@ extern(C) {
|
||||||
int libssh2_channel_wait_closed(LIBSSH2_CHANNEL *channel);
|
int libssh2_channel_wait_closed(LIBSSH2_CHANNEL *channel);
|
||||||
|
|
||||||
enum LIBSSH2_ERROR_EAGAIN = -37;
|
enum LIBSSH2_ERROR_EAGAIN = -37;
|
||||||
|
|
||||||
|
int libssh2_session_flag(LIBSSH2_SESSION*, int, int);
|
||||||
|
enum LIBSSH2_FLAG_SIGPIPE = 1;
|
||||||
|
enum LIBSSH2_FLAG_COMPRESS = 2;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
1
mangle.d
1
mangle.d
|
@ -1,3 +1,4 @@
|
||||||
|
/// a D mangler
|
||||||
module mangle;
|
module mangle;
|
||||||
|
|
||||||
import std.conv;
|
import std.conv;
|
||||||
|
|
2
mssql.d
2
mssql.d
|
@ -1,6 +1,6 @@
|
||||||
// NOTE: I haven't even tried to use this for a test yet!
|
// NOTE: I haven't even tried to use this for a test yet!
|
||||||
// It's probably godawful, if it works at all.
|
// It's probably godawful, if it works at all.
|
||||||
|
///
|
||||||
module arsd.mssql;
|
module arsd.mssql;
|
||||||
|
|
||||||
version(Windows):
|
version(Windows):
|
||||||
|
|
1
oauth.d
1
oauth.d
|
@ -1,3 +1,4 @@
|
||||||
|
///
|
||||||
module arsd.oauth;
|
module arsd.oauth;
|
||||||
|
|
||||||
import arsd.curl;
|
import arsd.curl;
|
||||||
|
|
1
png.d
1
png.d
|
@ -1,3 +1,4 @@
|
||||||
|
/// PNG file handling for color.d's Image interfaces
|
||||||
module arsd.png;
|
module arsd.png;
|
||||||
|
|
||||||
/// Easily reads a png file into a MemoryImage
|
/// Easily reads a png file into a MemoryImage
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
/// minimal libpq wrapper
|
||||||
module arsd.postgres;
|
module arsd.postgres;
|
||||||
pragma(lib, "pq");
|
pragma(lib, "pq");
|
||||||
|
|
||||||
|
|
1
sha.d
1
sha.d
|
@ -1,3 +1,4 @@
|
||||||
|
/// homemade sha 1 and sha2 impls. beware of bugs.
|
||||||
module arsd.sha;
|
module arsd.sha;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -52,7 +52,7 @@
|
||||||
compiler are all opt-in.
|
compiler are all opt-in.
|
||||||
|
|
||||||
simpledisplay.d's home base is on my arsd repo on Github. The file is:
|
simpledisplay.d's home base is on my arsd repo on Github. The file is:
|
||||||
$(L https://github.com/adamdruppe/arsd/blob/master/simpledisplay.d)
|
https://github.com/adamdruppe/arsd/blob/master/simpledisplay.d
|
||||||
|
|
||||||
simpledisplay is basically stable. I plan to refactor the internals,
|
simpledisplay is basically stable. I plan to refactor the internals,
|
||||||
and may add new features and fix bugs, but It do not expect to
|
and may add new features and fix bugs, but It do not expect to
|
||||||
|
@ -4416,8 +4416,8 @@ version(X11) {
|
||||||
version(with_eventloop) {
|
version(with_eventloop) {
|
||||||
import arsd.eventloop;
|
import arsd.eventloop;
|
||||||
static void eventListener(OsFileHandle fd) {
|
static void eventListener(OsFileHandle fd) {
|
||||||
this.mtLock();
|
//this.mtLock();
|
||||||
scope(exit) this.mtUnlock();
|
//scope(exit) this.mtUnlock();
|
||||||
while(XPending(display))
|
while(XPending(display))
|
||||||
doXNextEvent(display);
|
doXNextEvent(display);
|
||||||
}
|
}
|
||||||
|
|
3
sqlite.d
3
sqlite.d
|
@ -1,4 +1,4 @@
|
||||||
/*
|
/**
|
||||||
Compile with version=sqlite_extended_metadata_available
|
Compile with version=sqlite_extended_metadata_available
|
||||||
if your sqlite is compiled with the
|
if your sqlite is compiled with the
|
||||||
SQLITE_ENABLE_COLUMN_METADATA C-preprocessor symbol.
|
SQLITE_ENABLE_COLUMN_METADATA C-preprocessor symbol.
|
||||||
|
@ -8,7 +8,6 @@
|
||||||
use DataObjects, but you'll have to set up the mappings
|
use DataObjects, but you'll have to set up the mappings
|
||||||
manually without the extended metadata.)
|
manually without the extended metadata.)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
module arsd.sqlite;
|
module arsd.sqlite;
|
||||||
pragma(lib, "sqlite3");
|
pragma(lib, "sqlite3");
|
||||||
version(linux)
|
version(linux)
|
||||||
|
|
30
sslsocket.d
30
sslsocket.d
|
@ -1,4 +1,4 @@
|
||||||
/*
|
/**
|
||||||
This is CLIENT only at this point. Don't try to
|
This is CLIENT only at this point. Don't try to
|
||||||
bind/accept with these.
|
bind/accept with these.
|
||||||
|
|
||||||
|
@ -47,9 +47,18 @@ version(use_openssl) {
|
||||||
void SSL_free(SSL*);
|
void SSL_free(SSL*);
|
||||||
void SSL_CTX_free(SSL_CTX*);
|
void SSL_CTX_free(SSL_CTX*);
|
||||||
|
|
||||||
|
void SSL_set_verify(SSL*, int, void*);
|
||||||
|
enum SSL_VERIFY_NONE = 0;
|
||||||
|
|
||||||
SSL_METHOD* SSLv3_client_method();
|
SSL_METHOD* SSLv3_client_method();
|
||||||
|
SSL_METHOD* TLS_client_method();
|
||||||
|
SSL_METHOD* SSLv23_client_method();
|
||||||
|
|
||||||
|
void ERR_print_errors_fp(FILE*);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
import core.stdc.stdio;
|
||||||
|
|
||||||
shared static this() {
|
shared static this() {
|
||||||
SSL_library_init();
|
SSL_library_init();
|
||||||
OpenSSL_add_all_ciphers();
|
OpenSSL_add_all_ciphers();
|
||||||
|
@ -63,19 +72,26 @@ version(use_openssl) {
|
||||||
class OpenSslSocket : Socket {
|
class OpenSslSocket : Socket {
|
||||||
private SSL* ssl;
|
private SSL* ssl;
|
||||||
private SSL_CTX* ctx;
|
private SSL_CTX* ctx;
|
||||||
private void initSsl() {
|
private void initSsl(bool verifyPeer) {
|
||||||
ctx = SSL_CTX_new(SSLv3_client_method());
|
ctx = SSL_CTX_new(SSLv23_client_method());
|
||||||
assert(ctx !is null);
|
assert(ctx !is null);
|
||||||
|
|
||||||
ssl = SSL_new(ctx);
|
ssl = SSL_new(ctx);
|
||||||
|
if(!verifyPeer)
|
||||||
|
SSL_set_verify(ssl, SSL_VERIFY_NONE, null);
|
||||||
SSL_set_fd(ssl, this.handle);
|
SSL_set_fd(ssl, this.handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
@trusted
|
@trusted
|
||||||
override void connect(Address to) {
|
override void connect(Address to) {
|
||||||
super.connect(to);
|
super.connect(to);
|
||||||
if(SSL_connect(ssl) == -1)
|
if(SSL_connect(ssl) == -1) {
|
||||||
|
ERR_print_errors_fp(stderr);
|
||||||
|
int i;
|
||||||
|
printf("wtf\n");
|
||||||
|
scanf("%d\n", i);
|
||||||
throw new Exception("ssl connect");
|
throw new Exception("ssl connect");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@trusted
|
@trusted
|
||||||
|
@ -93,14 +109,14 @@ version(use_openssl) {
|
||||||
return receive(buf, SocketFlags.NONE);
|
return receive(buf, SocketFlags.NONE);
|
||||||
}
|
}
|
||||||
|
|
||||||
this(AddressFamily af, SocketType type = SocketType.STREAM) {
|
this(AddressFamily af, SocketType type = SocketType.STREAM, bool verifyPeer = true) {
|
||||||
super(af, type);
|
super(af, type);
|
||||||
initSsl();
|
initSsl(verifyPeer);
|
||||||
}
|
}
|
||||||
|
|
||||||
this(socket_t sock, AddressFamily af) {
|
this(socket_t sock, AddressFamily af) {
|
||||||
super(sock, af);
|
super(sock, af);
|
||||||
initSsl();
|
initSsl(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
~this() {
|
~this() {
|
||||||
|
|
|
@ -1,11 +1,12 @@
|
||||||
// stb_truetype.h - v0.6c - public domain
|
/// stb_truetype.h - v0.6c - public domain
|
||||||
// authored from 2009-2012 by Sean Barrett / RAD Game Tools
|
/// authored from 2009-2012 by Sean Barrett / RAD Game Tools
|
||||||
//
|
//
|
||||||
// http://nothings.org/stb/stb_truetype.h
|
// http://nothings.org/stb/stb_truetype.h
|
||||||
//
|
//
|
||||||
// port to D by adam d. ruppe. see the link above for more info about the lib and real author.
|
// port to D by adam d. ruppe. see the link above for more info about the lib and real author.
|
||||||
|
|
||||||
// here's some D convenience functions
|
// here's some D convenience functions
|
||||||
|
module stb_truetype;
|
||||||
|
|
||||||
struct TtfFont {
|
struct TtfFont {
|
||||||
stbtt_fontinfo font;
|
stbtt_fontinfo font;
|
||||||
|
|
12
terminal.d
12
terminal.d
|
@ -1516,13 +1516,15 @@ struct RealTimeConsoleInput {
|
||||||
FD_ZERO(&fs);
|
FD_ZERO(&fs);
|
||||||
|
|
||||||
FD_SET(fdIn, &fs);
|
FD_SET(fdIn, &fs);
|
||||||
select(fdIn + 1, &fs, null, null, &tv);
|
if(select(fdIn + 1, &fs, null, null, &tv) == -1) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
return FD_ISSET(fdIn, &fs);
|
return FD_ISSET(fdIn, &fs);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private bool anyInput_internal() {
|
/* private */ bool anyInput_internal() {
|
||||||
if(inputQueue.length || timedCheckForInput(0))
|
if(inputQueue.length || timedCheckForInput(0))
|
||||||
return true;
|
return true;
|
||||||
version(Posix)
|
version(Posix)
|
||||||
|
@ -3351,6 +3353,9 @@ version(Windows) {
|
||||||
|
|
||||||
|
|
||||||
struct ScrollbackBuffer {
|
struct ScrollbackBuffer {
|
||||||
|
|
||||||
|
bool demandsAttention;
|
||||||
|
|
||||||
this(string name) {
|
this(string name) {
|
||||||
this.name = name;
|
this.name = name;
|
||||||
}
|
}
|
||||||
|
@ -3640,6 +3645,8 @@ struct ScrollbackBuffer {
|
||||||
case InputEvent.Type.KeyboardEvent:
|
case InputEvent.Type.KeyboardEvent:
|
||||||
auto ev = e.keyboardEvent;
|
auto ev = e.keyboardEvent;
|
||||||
|
|
||||||
|
demandsAttention = false;
|
||||||
|
|
||||||
switch(ev.which) {
|
switch(ev.which) {
|
||||||
case KeyboardEvent.Key.UpArrow:
|
case KeyboardEvent.Key.UpArrow:
|
||||||
scrollUp();
|
scrollUp();
|
||||||
|
@ -3660,6 +3667,7 @@ struct ScrollbackBuffer {
|
||||||
case InputEvent.Type.MouseEvent:
|
case InputEvent.Type.MouseEvent:
|
||||||
auto ev = e.mouseEvent;
|
auto ev = e.mouseEvent;
|
||||||
if(ev.x >= x && ev.x < x + width && ev.y >= y && ev.y < y + height) {
|
if(ev.x >= x && ev.x < x + width && ev.y >= y && ev.y < y + height) {
|
||||||
|
demandsAttention = false;
|
||||||
// it is inside our box, so do something with it
|
// it is inside our box, so do something with it
|
||||||
auto mx = ev.x - x;
|
auto mx = ev.x - x;
|
||||||
auto my = ev.y - y;
|
auto my = ev.y - y;
|
||||||
|
|
3
web.d
3
web.d
|
@ -1,3 +1,4 @@
|
||||||
|
/// magic web wrapper
|
||||||
module arsd.web;
|
module arsd.web;
|
||||||
|
|
||||||
// it would be nice to be able to add meta info to a returned envelope
|
// it would be nice to be able to add meta info to a returned envelope
|
||||||
|
@ -3594,7 +3595,7 @@ struct TemplateFilters {
|
||||||
|
|
||||||
switch(word[$ - 1]) {
|
switch(word[$ - 1]) {
|
||||||
case 's':
|
case 's':
|
||||||
case 'a', 'e', 'i', 'o', 'u':
|
case 'a', 'i', 'o', 'u':
|
||||||
return word ~ "es";
|
return word ~ "es";
|
||||||
case 'f':
|
case 'f':
|
||||||
return word[0 .. $-1] ~ "ves";
|
return word[0 .. $-1] ~ "ves";
|
||||||
|
|
Loading…
Reference in New Issue