/******************************************************************************* * libresapi/api/StateTokenServer.cpp * * * * LibResAPI: API for local socket server * * * * Copyright 2018 by Retroshare Team * * * * This program is free software: you can redistribute it and/or modify * * it under the terms of the GNU Affero General Public License as * * published by the Free Software Foundation, either version 3 of the * * License, or (at your option) any later version. * * * * This program is distributed in the hope that it will be useful, * * but WITHOUT ANY WARRANTY; without even the implied warranty of * * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * * GNU Affero General Public License for more details. * * * * You should have received a copy of the GNU Affero General Public License * * along with this program. If not, see . * * * *******************************************************************************/ #include "StateTokenServer.h" #include namespace resource_api { // maybe it would be good to make this part of state token or friend, to be able to directly access the value StreamBase& operator <<(StreamBase& left, KeyValueReference kv) { if(left.serialise()) { // have to make a variable, to be able to pass it by reference // (cant pass return value of a function by refernce to another function) int value = kv.value.getValue(); left << makeKeyValueReference(kv.key, value); } else { int value; left << makeKeyValueReference(kv.key, value); kv.value = StateToken(value); } return left; } StreamBase& operator<<(StreamBase& left, StateToken& token) { if(left.serialise()) { // have to make a variable, to be able to pass it by reference // (cant pass return value of a function by refernce to another function) int value = token.getValue(); left << value; } else { int value; left << value; token = StateToken(value); } return left; } bool operator==(const StateToken& left, const StateToken& right) { if(left.getValue() == right.getValue()) { return true; } else { return false; } } StateTokenServer::StateTokenServer(): mMtx("StateTokenServer mMtx"), mNextToken(1), mClientsMtx("StateTokenServer mClientsMtx") { addResourceHandler("*", this, &StateTokenServer::handleWildcard); } StateToken StateTokenServer::getNewToken() { RS_STACK_MUTEX(mMtx); // ********** LOCKED ********** return locked_getNewToken(); } void StateTokenServer::discardToken(StateToken token) { RS_STACK_MUTEX(mMtx); // ********** LOCKED ********** locked_discardToken(token); } void StateTokenServer::replaceToken(StateToken &token) { RS_STACK_MUTEX(mMtx); // ********** LOCKED ********** locked_discardToken(token); token = locked_getNewToken(); } void StateTokenServer::registerTickClient(Tickable *c) { // extra service: tick it to let it init its ticking stuff c->tick(); // avoid double registration unregisterTickClient(c); RS_STACK_MUTEX(mClientsMtx); // ********** LOCKED ********** mTickClients.push_back(c); } void StateTokenServer::unregisterTickClient(Tickable *c) { RS_STACK_MUTEX(mClientsMtx); // ********** LOCKED ********** std::vector::iterator vit = std::find(mTickClients.begin(), mTickClients.end(), c); if(vit != mTickClients.end()) mTickClients.erase(vit); } void StateTokenServer::handleWildcard(Request &req, Response &resp) { { RS_STACK_MUTEX(mClientsMtx); // ********** LOCKED ********** for(std::vector::iterator vit = mTickClients.begin(); vit != mTickClients.end(); ++vit) { (*vit)->tick(); } } RS_STACK_MUTEX(mMtx); // ********** LOCKED ********** // want to lookpup many tokens at once, return a list of invalid tokens // TODO: make generic list serialiser/deserialiser resp.mDataStream.getStreamToMember(); while(req.mStream.hasMore()) { StateToken token; req.mStream << token; // lookup if token is valid if(std::find(mValidTokens.begin(), mValidTokens.end(), token) == mValidTokens.end()) { // if invalid, add to response list resp.mDataStream << token; } } resp.setOk(); } StateToken StateTokenServer::locked_getNewToken() { StateToken token(mNextToken); mValidTokens.push_back(token); mNextToken++; if(mNextToken == 0) // 0 is a reserved value, don't ever use it mNextToken = 1; return token; } void StateTokenServer::locked_discardToken(StateToken token) { std::vector::iterator toDelete = std::find(mValidTokens.begin(), mValidTokens.end(), token); if(toDelete != mValidTokens.end()) { mValidTokens.erase(toDelete); } } } // namespace resource_api