mirror of https://github.com/adamdruppe/arsd.git
sync stuff
This commit is contained in:
parent
94c474597d
commit
88c19d7355
145
rpc.d
145
rpc.d
|
@ -1,6 +1,13 @@
|
||||||
module arsd.rpc;
|
module arsd.rpc;
|
||||||
|
|
||||||
/+ //example usage
|
/*
|
||||||
|
FIXME:
|
||||||
|
1) integrate with arsd.eventloop
|
||||||
|
2) make it easy to use with other processes; pipe to a process and talk to it that way. perhaps with shared memory too?
|
||||||
|
3) extend the serialization capabilities
|
||||||
|
*/
|
||||||
|
|
||||||
|
///+ //example usage
|
||||||
interface ExampleNetworkFunctions {
|
interface ExampleNetworkFunctions {
|
||||||
string sayHello(string name);
|
string sayHello(string name);
|
||||||
int add(int a, int b);
|
int add(int a, int b);
|
||||||
|
@ -52,12 +59,20 @@ void main(string[] args) {
|
||||||
client.structTest(S1(20, "cool!"), (a) { writeln(a.name, " -- ", a.number); }, null);
|
client.structTest(S1(20, "cool!"), (a) { writeln(a.name, " -- ", a.number); }, null);
|
||||||
client.die(delegate () { writeln("shouldn't happen"); }, delegate(a) { writeln(a); });
|
client.die(delegate () { writeln("shouldn't happen"); }, delegate(a) { writeln(a); });
|
||||||
client.eventLoop();
|
client.eventLoop();
|
||||||
|
|
||||||
|
/*
|
||||||
|
auto client = makeNetworkClient!(ExampleNetworkFunctions, false)("localhost", 5005);
|
||||||
|
writeln(client.sayHello("whoa"));
|
||||||
|
writeln(client.add(1, 2));
|
||||||
|
client.die();
|
||||||
|
writeln(client.add(1, 2));
|
||||||
|
*/
|
||||||
} else {
|
} else {
|
||||||
auto server = new ExampleServer(5005);
|
auto server = new ExampleServer(5005);
|
||||||
server.eventLoop();
|
server.eventLoop();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
+/
|
//+/
|
||||||
|
|
||||||
mixin template NetworkServer(Interface) {
|
mixin template NetworkServer(Interface) {
|
||||||
import std.socket;
|
import std.socket;
|
||||||
|
@ -260,22 +275,27 @@ final public inout(ubyte)[] deserializeInto(T)(inout(ubyte)[] buffer, ref T s) {
|
||||||
return buffer;
|
return buffer;
|
||||||
}
|
}
|
||||||
|
|
||||||
mixin template NetworkClient(Interface) {
|
mixin template NetworkClient(Interface, bool useAsync = true) {
|
||||||
private static string createClass() {
|
private static string createClass() {
|
||||||
// this doesn't actually inherit from the interface because
|
// this doesn't actually inherit from the interface because
|
||||||
// the return value needs to be handled async
|
// the return value needs to be handled async
|
||||||
string code;// = `final class Class /*: ` ~ Interface.stringof ~ `*/ {`;
|
string code;// = `final class Class /*: ` ~ Interface.stringof ~ `*/ {`;
|
||||||
code ~= "\n\timport std.socket;";
|
code ~= "\n\timport std.socket;";
|
||||||
code ~= "\n\tprivate Socket socket;";
|
code ~= "\n\tprivate Socket socket;";
|
||||||
code ~= "\n\tprivate void delegate(const(ubyte)[] buffer)[uint] onSuccesses;";
|
if(useAsync) {
|
||||||
code ~= "\n\tprivate void delegate(const(ubyte)[] buffer)[uint] onErrors;";
|
code ~= "\n\tprivate void delegate(const(ubyte)[] buffer)[uint] onSuccesses;";
|
||||||
|
code ~= "\n\tprivate void delegate(const(ubyte)[] buffer)[uint] onErrors;";
|
||||||
|
}
|
||||||
code ~= "\n\tprivate uint lastSequenceNumber;";
|
code ~= "\n\tprivate uint lastSequenceNumber;";
|
||||||
code ~= q{
|
code ~= q{
|
||||||
private this(string host, ushort port) {
|
private this(string host, ushort port) {
|
||||||
this.socket = new TcpSocket();
|
this.socket = new TcpSocket();
|
||||||
this.socket.connect(new InternetAddress(host, port));
|
this.socket.connect(new InternetAddress(host, port));
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if(useAsync)
|
||||||
|
code ~= q{
|
||||||
final public void eventLoop() {
|
final public void eventLoop() {
|
||||||
ubyte[4096] buffer;
|
ubyte[4096] buffer;
|
||||||
bool open = true;
|
bool open = true;
|
||||||
|
@ -326,10 +346,19 @@ mixin template NetworkClient(Interface) {
|
||||||
code ~= "\n\tpublic:\n";
|
code ~= "\n\tpublic:\n";
|
||||||
|
|
||||||
foreach(memIdx, member; __traits(allMembers, Interface)) {
|
foreach(memIdx, member; __traits(allMembers, Interface)) {
|
||||||
alias mem = PassThrough!(__traits(getMember, Interface, member));
|
|
||||||
code ~= "\t\tfinal void " ~ member ~ "(";
|
|
||||||
bool hadArgument = false;
|
|
||||||
import std.traits;
|
import std.traits;
|
||||||
|
alias mem = PassThrough!(__traits(getMember, Interface, member));
|
||||||
|
string type;
|
||||||
|
if(useAsync)
|
||||||
|
type = "void";
|
||||||
|
else {
|
||||||
|
static if(is(ReturnType!mem == void))
|
||||||
|
type = "void";
|
||||||
|
else
|
||||||
|
type = (ReturnType!mem).stringof;
|
||||||
|
}
|
||||||
|
code ~= "\t\tfinal "~type~" " ~ member ~ "(";
|
||||||
|
bool hadArgument = false;
|
||||||
import std.conv;
|
import std.conv;
|
||||||
// arguments
|
// arguments
|
||||||
foreach(i, arg; ParameterTypeTuple!mem) {
|
foreach(i, arg; ParameterTypeTuple!mem) {
|
||||||
|
@ -340,19 +369,22 @@ mixin template NetworkClient(Interface) {
|
||||||
hadArgument = true;
|
hadArgument = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(hadArgument)
|
if(useAsync) {
|
||||||
code ~= ", ";
|
if(hadArgument)
|
||||||
|
code ~= ", ";
|
||||||
|
|
||||||
static if(is(ReturnType!mem == void))
|
static if(is(ReturnType!mem == void))
|
||||||
code ~= "void delegate() onSuccess";
|
code ~= "void delegate() onSuccess";
|
||||||
else
|
else
|
||||||
code ~= "void delegate("~(ReturnType!mem).stringof~") onSuccess";
|
code ~= "void delegate("~(ReturnType!mem).stringof~") onSuccess";
|
||||||
code ~= ", ";
|
code ~= ", ";
|
||||||
code ~= "void delegate(Throwable) onError";
|
code ~= "void delegate(Throwable) onError";
|
||||||
|
}
|
||||||
code ~= ") {\n";
|
code ~= ") {\n";
|
||||||
|
code ~= "auto seq = ++lastSequenceNumber;";
|
||||||
|
if(useAsync)
|
||||||
code ~= q{
|
code ~= q{
|
||||||
#line 252
|
#line 252
|
||||||
auto seq = ++lastSequenceNumber;
|
|
||||||
onSuccesses[seq] = (const(ubyte)[] buffer) {
|
onSuccesses[seq] = (const(ubyte)[] buffer) {
|
||||||
onSuccesses.remove(seq);
|
onSuccesses.remove(seq);
|
||||||
onErrors.remove(seq);
|
onErrors.remove(seq);
|
||||||
|
@ -381,7 +413,9 @@ mixin template NetworkClient(Interface) {
|
||||||
if(onError !is null)
|
if(onError !is null)
|
||||||
onError(t);
|
onError(t);
|
||||||
};
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
code ~= q{
|
||||||
#line 283
|
#line 283
|
||||||
ubyte[4096] bufferBase;
|
ubyte[4096] bufferBase;
|
||||||
auto buffer = bufferBase[12 .. $]; // leaving room for size, func number, and seq number
|
auto buffer = bufferBase[12 .. $]; // leaving room for size, func number, and seq number
|
||||||
|
@ -402,10 +436,77 @@ mixin template NetworkClient(Interface) {
|
||||||
// FIXME: what if it doesn't all send at once?
|
// FIXME: what if it doesn't all send at once?
|
||||||
code ~= "\t\t\tsocket.send(bufferBase[0 .. 12 + used]);\n";
|
code ~= "\t\t\tsocket.send(bufferBase[0 .. 12 + used]);\n";
|
||||||
//code ~= `writeln("sending ", bufferBase[0 .. 12 + used]);`;
|
//code ~= `writeln("sending ", bufferBase[0 .. 12 + used]);`;
|
||||||
|
|
||||||
|
if(!useAsync)
|
||||||
|
code ~= q{
|
||||||
|
ubyte[4096] dbuffer;
|
||||||
|
bool open = true;
|
||||||
|
static if(is(typeof(return) == void)) {
|
||||||
|
|
||||||
|
} else
|
||||||
|
typeof(return) returned;
|
||||||
|
|
||||||
|
auto gotNum = socket.receive(dbuffer);
|
||||||
|
if(gotNum == 0) {
|
||||||
|
open = false;
|
||||||
|
throw new Exception("connection closed");
|
||||||
|
}
|
||||||
|
while(gotNum < 9) {
|
||||||
|
auto g2 = socket.receive(dbuffer[gotNum .. $]);
|
||||||
|
if(g2 == 0) {
|
||||||
|
open = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
gotNum += g2;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto got = dbuffer[0 .. gotNum];
|
||||||
|
another:
|
||||||
|
uint length;
|
||||||
|
uint success;
|
||||||
|
got = deserializeInto(got, length);
|
||||||
|
got = deserializeInto(got, seq);
|
||||||
|
got = deserializeInto(got, success);
|
||||||
|
auto more = got[length .. $];
|
||||||
|
|
||||||
|
if(got.length >= length) {
|
||||||
|
if(success) {
|
||||||
|
/*
|
||||||
|
auto s = (seq in onSuccesses);
|
||||||
|
if(s !is null && *s !is null)
|
||||||
|
(*s)(got);
|
||||||
|
*/
|
||||||
|
static if(is(typeof(return) == void)) {
|
||||||
|
} else {
|
||||||
|
got = deserializeInto(got, returned);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
/*
|
||||||
|
auto s = (seq in onErrors);
|
||||||
|
if(s !is null && *s !is null)
|
||||||
|
(*s)(got);
|
||||||
|
*/
|
||||||
|
auto t = new Throwable("");
|
||||||
|
got = deserializeInto(got, t.msg);
|
||||||
|
got = deserializeInto(got, t.file);
|
||||||
|
got = deserializeInto(got, t.line);
|
||||||
|
throw t;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(more.length) {
|
||||||
|
got = more;
|
||||||
|
goto another;
|
||||||
|
}
|
||||||
|
static if(is(typeof(return) == void)) {
|
||||||
|
|
||||||
|
} else
|
||||||
|
return returned;
|
||||||
|
};
|
||||||
|
|
||||||
code ~= "}\n";
|
code ~= "}\n";
|
||||||
code ~= "\n";
|
code ~= "\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
//code ~= `}`;
|
//code ~= `}`;
|
||||||
return code;
|
return code;
|
||||||
}
|
}
|
||||||
|
@ -414,6 +515,14 @@ mixin template NetworkClient(Interface) {
|
||||||
mixin(createClass());
|
mixin(createClass());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
auto makeNetworkClient(Interface, bool useAsync = true)(string host, ushort port) {
|
||||||
|
class Thing {
|
||||||
|
mixin NetworkClient!(Interface, useAsync);
|
||||||
|
}
|
||||||
|
|
||||||
|
return new Thing(host, port);
|
||||||
|
}
|
||||||
|
|
||||||
// the protocol is:
|
// the protocol is:
|
||||||
/*
|
/*
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue