110 const std::string& message) :
116 memcpy(this->
buffer, static_cast<const JPrefix_t*>(
this),
sizeof(
JPrefix_t));
118 memcpy(this->
buffer +
sizeof(
JPrefix_t), message.data(), message.size());
129 static_cast<JPrefix_t&
>(*this) = message;
354 for (istringstream is(subscription); is >> c >> tag; ) {
414 queue.push_back(message);
428 for (std::deque<JDispatch>::iterator i =
queue.begin(); i !=
queue.end(); ) {
472 for (
iterator i = this->begin(); i != this->end(); ++i) {
483 for (
iterator i = this->begin(); i != this->end(); ++i) {
499 return out <<
"(" << message.
getTag() <<
"," << message.
size() <<
")";
555 int main(
int argc,
char* argv[])
580 catch(
const exception &error) {
581 FATAL(error.what() << endl);
585 JServerSocket server(port, backlog);
587 JClientList clientList;
590 DEBUG(
"Port " << setw(10) << port << endl);
598 select.setReaderMask(server);
600 for (JClientList::iterator client = clientList.begin(); client != clientList.end(); ++client) {
602 if (!client->in.isReady()) {
603 select.setReaderMask(*client);
606 if (client->out.isReset()) {
608 if (!client->queue.empty()) {
610 DEBUG(
"Client" << *client <<
".set" << client->queue.front() << endl);
612 client->out.set(client->queue.front());
613 client->decrementRequest();
615 select.setWriterMask(*client);
618 }
else if (client->out.isBusy()) {
620 select.setWriterMask(*client);
624 if (select(timeout_us) > 0) {
626 for (JClientList::iterator client = clientList.begin(); client != clientList.end(); ) {
630 if (select.hasReaderMask(*client)) {
635 catch(
const exception& error) {
637 ERROR(
"Remove (3) client" << *client <<
"<" << client->getNickname() <<
">: " << error.what() << endl);
639 if (client->getNickname() !=
"") {
640 clientList.add(JDispatch(
DISPTAG_Died, client->getNickname()));
645 client = clientList.erase(client);
650 DEBUG(
"Client" << *client <<
".read" << static_cast<const JSocketInputBuffer&>(client->in) << endl);
653 if (client->in.isReady()) {
655 DEBUG(
"Message" << *client <<
' ' << client->in.prefix.c_str() <<
' ' << client->in.size() << endl);
665 client->setSubscription(
string(client->in.getRemainingData(), client->in.getRemainingSize()));
667 }
else if (client->in.prefix.getTag() ==
DISPTAG_MyId) {
669 client->setNickname(
string(client->in.getRemainingData(), client->in.getRemainingSize()));
671 clientList.add(JDispatch(
DISPTAG_Born, client->getNickname()));
673 }
else if (client->in.prefix.getTag() ==
DISPTAG_Gime) {
675 client->incrementRequest();
679 client->setRequestAll();
683 string nick_name(client->in.getRemainingData(), client->in.getRemainingSize());
686 for (JClientList::iterator i = clientList.begin(); i != clientList.end(); ++i) {
687 if (i->getNickname() == nick_name) {
688 buffer +=
" " + i->getHostname();
692 JControlHost socket(*client);
694 socket.PutFullString(
DISPTAG_WhereIs, buffer.substr(buffer.empty() ? 0 : 1));
696 DEBUG(
"Remove (1) client" << *client << endl);
700 client = clientList.erase(client);
712 clientList.add(JDispatch(client->in.prefix, client->in.data()));
725 if (select.hasWriterMask(*client)) {
729 DEBUG(
"Client" << *client <<
".write" << static_cast<const JSocketStatus&>(client->out) << endl);
731 if (client->out.isReady()) {
733 client->queue.pop_front();
739 catch(
const exception& error) {
741 DEBUG(
"Remove (2) client" << *client <<
"<" << client->getNickname() <<
">: " << error.what() << endl);
743 if (client->getNickname() !=
"") {
744 clientList.add(JDispatch(
DISPTAG_Died, client->getNickname()));
749 client = clientList.erase(client);
753 if (select.hasReaderMask(server)) {
757 socket.accept(server.getFileDescriptor());
762 socket.setKeepAlive (
true);
763 socket.setNonBlocking(
true);
765 DEBUG(
"New client" << socket << endl);
767 clientList.push_back(JClient(socket));
JSocketNonblockingWriter out
writer for outgoing messages
bool setSubscription(const std::string &subscription)
Set subcription.
Utility class to parse command line options.
static const JTag DISPTAG_Subscribe("_Subscri")
Special ControlHost tags.
std::set< JTag > subscriptionAll
int getSizeOfPacket(const KM3NETDAQ::JDAQAbstractPreamble &preamble)
static const int DISPATCH_PORT
Default ControlHost port.
JDispatch & operator=(const JDispatch &message)
Assignment operator.
static const JTag DISPTAG_Died("Died")
Data structure of a ControlHost message.
JSocketBuffer< const char > JSocketOutputBuffer
void setRequestAll()
Set no request.
ControlHost client manager.
std::deque< JDispatch > queue
queue for outgoing messages
void release()
Release memory.
Auxiliary class for non-blocking socket I/O.
List of ControlHost client managers.
std::set< JTag > subscriptionAny
static const JTag DISPTAG_MyId("_MyId")
void attach(const JSharedCounter &object)
Attach this counter to given shared counter object.
void setNickname(const std::string &nick_name)
Set nick name.
static const JTag DISPTAG_WhereIs("_WhereIs")
bool checkRequest() const
Check request.
bool checkSubscription(const JPrefix_t &prefix) const
Check subscription for given prefix.
void setSize(const int length)
Set size.
JClient()
Default constructor.
const std::string & getNickname() const
Get nick name.
void decrementRequest()
Decrement request by one.
bool checkSubscriptionAll(const JPrefix_t &prefix) const
Check subscription for given prefix.
static long long int MEMORY_LIMIT
Limit size of data [Bytes].
int getSize() const
Get size.
JClientList()
Default constructor.
static void release(T *p)
Release memory.
bool checkSubscriptionAny(const JPrefix_t &prefix) const
Check subscription for given prefix.
int getFileDescriptor() const
Get file descriptor.
JSocketInputChannel_t in
reader for incoming messages
int getCounter() const
Get number of I/O attempts.
std::ostream & operator<<(std::ostream &out, const JDispatch &message)
Print message.
Non-blocking socket writer.
void incrementRequest()
Increment request by one.
#define make_field(A,...)
macro to convert parameter to JParserTemplateElement object
bool isReady() const
Check ready status.
void add(const JDispatch &message)
Add message to client queues depending on subscription of each client.
Memory management class for create/release policy based on malloc()/free().
Exception for failure of memory allocation.
static const size_t buffer_size
JDispatch()
Default constructor.
void drop()
Drop all messages for which the client has not the 'all' subscription.
JDispatch(const JPrefix_t &prefix, const char *data)
Constructor.
JDispatch(const JDispatch &message)
Copy constructor.
void create()
Allocate memory.
void drop()
Drop all messages from client queues for which the client has not the 'all' subscription.
Exception for ControlHost.
JClient(const JSocket &socket)
Constructor.
const JTag & getTag() const
Get tag.
static bool maybe_special(const JTag &tag)
Check special ControlHost tags.
General purpose messaging.
void initialise()
Initialise counter.
static const JTag DISPTAG_Born("Born")
static long long int MEMORY_TOTAL
Total size of data [Bytes].
JMalloc< char > JMemory_t
static unsigned int QUEUE_LIMIT
Maximum number of messages in queue.
const char * data() const
Get data.
Utility class to parse command line options.
void add(const JDispatch &message)
Add message to client queues depending on subscription of each client.
unsigned long long int getRAM()
Get RAM of this CPU.
static const JTag DISPTAG_Gime("_Gime")
int getSize() const
Get size of pending data.
JSocketInputChannel< JPrefix_t > JSocketInputChannel_t
void setSizeOfPacket(const int size, JPrefix_t &prefix)
Set total size of internet packet.
static const JTag DISPTAG_Always("_Always")
Base class for memory management.
Auxiliary class for non-blocking socket I/O.
static T * create()
Create object in memory.
bool isBusy() const
Check busy status.
int size() const
Get size.
JDispatch(const JTag &tag, const std::string &message)
Constructor.
#define DEBUG(A)
Message macros.
int main(int argc, char *argv[])