/*************************************************************************
* 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);
}