/** * Engine * * Contains the core components of the tasky * library, this is effectively the entry * point to the library */ module tasky.engine; import eventy.engine : EvEngine = Engine; import eventy.event : Event; import tasky.jobs : Descriptor; import tristanable; import std.socket : Socket; import core.thread : Thread; public final class Engine : Thread { /** * Tristanable sub-system */ private Manager tmanager; /** * Eventy sub-system */ private EvEngine evEngine; this(Socket socket) { /* Set the worker function */ super(&worker); /* TODO: Check for exceptions */ /* Create a new event engine */ evEngine = new EvEngine(); evEngine.start(); /* TODO: Check for exceptions */ /* Create a new tristanable manager */ tmanager = new Manager(socket); /* Start the loop */ start(); } public class TaskyEvent : Event { private byte[] payload; this(ulong descID, byte[] payload) { super(descID); this.payload = payload; } } /** * Worker thread function which checks the tristanable * queues for whichever has messages on them and then * dispatches a job-response for them via eventy */ private void worker() { ulong got = 0; while(true) { //writeln("WHITE BOY SUMMER"); /* TODO: Get all tristanable queues */ Queue[] tQueues = tmanager.getQueues(); // writeln("Got: ", got); foreach(Queue tQueue; tQueues) { // writeln("Check queue: ", tQueue); /* Descriptor ID */ ulong descID = tQueue.getTag(); /* Check if the queue has mail */ /* TODO: Different discplines here, full-exhaust or round robin queue */ if(tQueue.poll()) { /** * Dequeue the data item and push * event into the event loop containing * it */ QueueItem data = tQueue.dequeue(); evEngine.push(new TaskyEvent(descID, data.getData())); } } /* TODO: Use queue ID to match to descriptor id for later job dispatch */ /* TODO: Per each queue */ /* TODO: Yield away somehow */ import core.thread : dur; // sleep(dur!("msecs")(500)); } } /** * Register a Descriptor with tasky */ public void registerDescriptor(Descriptor desc) { /* Add a queue based on the descriptor ID */ evEngine.addQueue(desc.getDescriptorClass()); /* Add a signal handler that handles said descriptor ID */ evEngine.addSignalHandler(desc); /* Create a new queue for this Job */ Queue tQueue = new Queue(desc.getDescriptorClass()); /* Add the Queue to tristanable */ tmanager.addQueue(tQueue); } unittest { /* Results array for unit testing */ bool[4] results; import std.conv : to; import core.thread : dur; import std.string : cmp; /* Job type */ Descriptor jobType = new class Descriptor { public override void handler(Event e) { import std.stdio : writeln; writeln("Event id ", e.id); TaskyEvent eT = cast(TaskyEvent)e; string data = cast(string)eT.payload; writeln(data); if(cmp(data, "Hello 1") == 0) { results[0] = true; } else if(cmp(data, "Hello 2") == 0) { results[1] = true; } } }; ulong jobTypeDI = jobType.getDescriptorClass; ulong job2C = 0; /* Job type */ Descriptor jobType2 = new class Descriptor { public override void handler(Event e) { import std.stdio : writeln; writeln("Event id ", e.id); writeln("OTHER event type"); TaskyEvent eT = cast(TaskyEvent)e; string data = cast(string)eT.payload; writeln(data); // job2C++; // assert(cmp(cast(string)eT.payload, "")) if(cmp(data, "Bye-bye! 3") == 0) { results[2] = true; } else if(cmp(data, "Bye-bye! 4") == 0) { results[3] = true; } } }; ulong jobTypeDI2 = jobType2.getDescriptorClass; import std.socket; import std.stdio; Socket serverSocket = new Socket(AddressFamily.INET6, SocketType.STREAM, ProtocolType.TCP); serverSocket.bind(parseAddress("::1", 0)); Address serverAddress = serverSocket.localAddress(); Thread serverThread = new class Thread { this() { super(&worker); serverSocket.listen(0); } public void worker() { Socket clientSocket = serverSocket.accept(); sleep(dur!("seconds")(2)); import tristanable.encoding : DataMessage, encodeForSend; DataMessage dMesg = new DataMessage(jobTypeDI, cast(byte[])"Hello 1"); writeln("Server send 1: ", clientSocket.send(encodeForSend(dMesg))); dMesg = new DataMessage(jobTypeDI, cast(byte[])"Hello 2"); writeln("Server send 2: ", clientSocket.send(encodeForSend(dMesg))); dMesg = new DataMessage(jobTypeDI2, cast(byte[])"Bye-bye! 3"); writeln("Server send 3: ", clientSocket.send(encodeForSend(dMesg))); dMesg = new DataMessage(jobTypeDI2, cast(byte[])"Bye-bye! 4"); writeln("Server send 4: ", clientSocket.send(encodeForSend(dMesg))); while(true) { } } }; serverThread.start(); Socket clientSocket = new Socket(AddressFamily.INET6, SocketType.STREAM, ProtocolType.TCP); clientSocket.connect(parseAddress("::1", to!(ushort)(serverAddress.toPortString()))); /* FIXME: Don't pass in null */ Engine e = new Engine(clientSocket); e.registerDescriptor(jobType); e.registerDescriptor(jobType2); while(!results[0] || !results[1] || !results[2] || !results[3]) { /* Check that the array has the correct values */ /* FIXME: Add timeout */ } /* TODO: Shutdown tasky here (shutdown eventy and tristanable) */ } }