/************************************************************************* * UrBackup - Client/Server backup system * Copyright (C) 2011-2016 Martin Raiber * * 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 "ServerIdentityMgr.h" #include "../Interface/Server.h" #include "../stringtools.h" #include #include #include "file_permissions.h" #include "../urbackupcommon/os_functions.h" const unsigned int ident_online_timeout=1*60*60*1000; //1h std::vector ServerIdentityMgr::identities; IMutex *ServerIdentityMgr::mutex=NULL; IFileServ *ServerIdentityMgr::filesrv=NULL; std::vector ServerIdentityMgr::new_identities; std::vector ServerIdentityMgr::session_identities; #ifdef _WIN32 const std::string server_ident_file="server_idents.txt"; const std::string server_session_ident_file="session_idents.txt"; const std::string server_new_ident_file="new_server_idents.txt"; #else const std::string server_ident_file="urbackup/server_idents.txt"; const std::string server_session_ident_file="urbackup/session_idents.txt"; const std::string server_new_ident_file="urbackup/new_server_idents.txt"; #endif void ServerIdentityMgr::init_mutex(void) { mutex=Server->createMutex(); } void ServerIdentityMgr::destroy_mutex(void) { Server->destroy(mutex); } void ServerIdentityMgr::addServerIdentity(const std::string &pIdentity, const SPublicKeys& pPublicKey) { IScopedLock lock(mutex); loadServerIdentities(); identities.push_back(SIdentity(pIdentity, pPublicKey)); if(pPublicKey.empty()) { filesrv->addIdentity("#I"+pIdentity+"#"); } writeServerIdentities(); } bool ServerIdentityMgr::checkServerSessionIdentity(const std::string &pIdentity, const std::string& endpoint) { IScopedLock lock(mutex); for(size_t i=0;igetTimeMS()-timegetTimeMS(); return true; } } return false; } bool ServerIdentityMgr::checkServerIdentity(const std::string &pIdentity) { IScopedLock lock(mutex); return std::find(identities.begin(), identities.end(), SIdentity(pIdentity))!=identities.end(); } void ServerIdentityMgr::loadServerIdentities(void) { IScopedLock lock(mutex); std::vector old_identities=identities; identities.clear(); std::string data=getFile(server_ident_file); int numl=linecount(data); for(int i=0;i<=numl;++i) { std::string l=trim(getline(i, data)); if(!l.empty() && l[0]=='#') { continue; } if(l.size()>5) { size_t hashpos = l.find("#"); SPublicKeys pubkeys; if(hashpos!=std::string::npos) { str_map params; ParseParamStrHttp(l.substr(hashpos+1), ¶ms); pubkeys = SPublicKeys(base64_decode_dash(params["pubkey"]), base64_decode_dash(params["pubkey_ecdsa409k1"])); l = l.substr(0, hashpos); } else { filesrv->addIdentity("#I"+l+"#"); } identities.push_back(SIdentity(l, pubkeys)); std::vector::iterator it=std::find(old_identities.begin(), old_identities.end(), SIdentity(l)); if(it!=old_identities.end()) { identities[identities.size()-1].onlinetime = it->onlinetime; } } } new_identities.clear(); data=getFile(server_new_ident_file); numl=linecount(data); for(int i=0;i<=numl;++i) { std::string l=trim(getline(i, data)); if(l.size()>5) { new_identities.push_back(l); } } std::vector old_session_identities=session_identities; session_identities.clear(); data=getFile(server_session_ident_file); numl=linecount(data); for(int i=0;i<=numl;++i) { std::string l=trim(getline(i, data)); if(l.size()>5) { size_t hashpos = l.find("#"); if(hashpos==std::string::npos) continue; str_map params; ParseParamStrHttp(l.substr(hashpos+1), ¶ms); l = l.substr(0, hashpos); SSessionIdentity session_ident(l, params["endpoint"], -1*Server->getTimeMS()); session_identities.push_back(session_ident); filesrv->addIdentity("#I"+l+"#"); std::vector::iterator it=std::find(old_session_identities.begin(), old_session_identities.end(), SSessionIdentity(session_ident)); if(it!=old_session_identities.end()) { session_identities[session_identities.size()-1].onlinetime = it->onlinetime; } } } } void ServerIdentityMgr::setFileServ(IFileServ *pFilesrv) { filesrv=pFilesrv; } size_t ServerIdentityMgr::numServerIdentities(void) { IScopedLock lock(mutex); return identities.size(); } bool ServerIdentityMgr::isNewIdentity(const std::string &pIdentity) { IScopedLock lock(mutex); if( std::find(new_identities.begin(), new_identities.end(), pIdentity)!=new_identities.end()) { return false; } else { new_identities.push_back(pIdentity); writeServerIdentities(); return true; } } void ServerIdentityMgr::writeServerIdentities(void) { IScopedLock lock(mutex); std::string idents; for(size_t i=0;igetTimeMS(); for(size_t i=0;i::iterator it = std::find(identities.begin(), identities.end(), SIdentity(pIdentity)); if(it!=identities.end()) { return it->publickeys; } return SPublicKeys(std::string(), std::string()); } bool ServerIdentityMgr::setPublicKeys( const std::string &pIdentity, const SPublicKeys &pPublicKeys ) { IScopedLock lock(mutex); std::vector::iterator it = std::find(identities.begin(), identities.end(), SIdentity(pIdentity)); if(it!=identities.end()) { it->publickeys = pPublicKeys; if(!pPublicKeys.empty()) { filesrv->removeIdentity(pIdentity); } writeServerIdentities(); return true; } return false; } bool ServerIdentityMgr::hasPublicKey( const std::string &pIdentity ) { IScopedLock lock(mutex); std::vector::iterator it = std::find(identities.begin(), identities.end(), SIdentity(pIdentity)); if(it!=identities.end()) { return !it->publickeys.empty(); } return false; } void ServerIdentityMgr::addSessionIdentity( const std::string &pIdentity, const std::string& endpoint ) { IScopedLock lock(mutex); SSessionIdentity session_ident(pIdentity, endpoint, Server->getTimeMS()); session_identities.push_back(session_ident); filesrv->addIdentity("#I"+pIdentity+"#"); writeSessionIdentities(); } void ServerIdentityMgr::writeSessionIdentities() { IScopedLock lock(mutex); const size_t max_session_identities=1000; size_t start=0; if(session_identities.size()>max_session_identities) { start=session_identities.size()-max_session_identities; } size_t written = 0; std::string idents; for(size_t i=session_identities.size(); i-- >0;) { int64 time = session_identities[i].onlinetime; if(time<0) time*=-1; if(Server->getTimeMS()-time=max_session_identities) { break; } } } write_file_admin_atomic(idents, server_session_ident_file); } bool ServerIdentityMgr::write_file_admin_atomic(const std::string & data, const std::string & fn) { if (!write_file_only_admin(data, fn + ".new")) { return false; } { std::auto_ptr f(Server->openFile(fn + ".new", MODE_RW)); if (f.get() != NULL) f->Sync(); } return os_rename_file(fn + ".new", fn); }