/*************************************************************************
* 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 "FileServ.h"
#include "map_buffer.h"
#include "../stringtools.h"
#include "../Interface/Server.h"
#include "CClientThread.h"
#include "../Interface/ThreadPool.h"
#include
#include "CUDPThread.h"
#include "PipeSessions.h"
#include "PipeFileExt.h"
IMutex *FileServ::mutex=NULL;
std::vector FileServ::identities;
bool FileServ::pause=false;
std::map FileServ::script_mappings;
IFileServ::ITokenCallbackFactory* FileServ::token_callback_factory = NULL;
std::map FileServ::fn_redirects;
std::map FileServ::active_shares;
FileServ::IReadErrorCallback* FileServ::read_error_callback = NULL;
std::vector FileServ::read_error_files;
std::map, IFileServ::CbtHashFileInfo> FileServ::cbt_hash_files;
FileServ::FileServ(bool *pDostop, const std::string &pServername, THREADPOOL_TICKET serverticket, bool use_fqdn)
: servername(pServername), serverticket(serverticket)
{
dostop=pDostop;
if(servername.empty())
{
servername=getSystemServerName(use_fqdn);
}
}
FileServ::~FileServ(void)
{
delete dostop;
}
void FileServ::shareDir(const std::string &name, const std::string &path, const std::string& identity, bool allow_exec)
{
add_share_path(name, path, identity, allow_exec);
}
bool FileServ::removeDir(const std::string &name, const std::string& identity)
{
{
IScopedLock lock(mutex);
std::map, IFileServ::CbtHashFileInfo>::iterator it = cbt_hash_files.find(std::make_pair(name, identity));
if (it != cbt_hash_files.end())
{
Server->destroy(it->second.cbt_hash_file);
cbt_hash_files.erase(it);
}
}
return remove_share_path(name, identity);
}
void FileServ::stopServer(void)
{
*dostop=true;
Server->getThreadPool()->waitFor(serverticket);
}
std::string FileServ::getShareDir(const std::string &name, const std::string& identity)
{
bool allow_exec;
return map_file(name, identity, allow_exec, NULL);
}
void FileServ::addIdentity(const std::string &pIdentity)
{
IScopedLock lock(mutex);
if(std::find(identities.begin(), identities.end(), pIdentity)
== identities.end())
{
identities.push_back(pIdentity);
}
}
void FileServ::init_mutex(void)
{
mutex=Server->createMutex();
}
void FileServ::destroy_mutex(void)
{
Server->destroy(mutex);
}
bool FileServ::checkIdentity(const std::string &pIdentity)
{
IScopedLock lock(mutex);
for(size_t i=0;i* extra_buffer)
{
CClientThread cc(cp, NULL, extra_buffer);
cc();
}
bool FileServ::removeIdentity( const std::string &pIdentity )
{
IScopedLock lock(mutex);
std::vector::iterator it = std::find(identities.begin(), identities.end(), pIdentity);
if(it!=identities.end())
{
identities.erase(it);
return true;
}
else
{
return false;
}
}
bool FileServ::getExitInformation(const std::string& cmd, std::string& stderr_data, int& exit_code)
{
std::string pcmd;
bool allow_exec;
if (next(cmd, 0, "urbackup/TAR"))
{
std::string server_ident = getbetween("|", "|", cmd);
pcmd = getafter("urbackup/TAR|" + server_ident + "|", cmd);
std::string map_res = map_file(pcmd, std::string(), allow_exec, NULL);
if (map_res.empty())
{
return false;
}
}
else
{
pcmd = map_file(cmd, std::string(), allow_exec, NULL);
}
if (!allow_exec)
{
return false;
}
SExitInformation exit_info = PipeSessions::getExitInformation(pcmd);
if(exit_info.created==0)
{
return false;
}
else
{
stderr_data = exit_info.outdata;
exit_code = exit_info.rc;
return true;
}
}
void FileServ::addScriptOutputFilenameMapping(const std::string& script_output_fn, const std::string& script_fn, bool tar_file)
{
IScopedLock lock(mutex);
script_mappings[script_output_fn] = SScriptMapping(script_fn, tar_file, NULL);
}
std::string FileServ::mapScriptOutputNameToScript(const std::string& script_fn, bool& tar_file, IPipeFile*& pipe_file)
{
IScopedLock lock(mutex);
std::map::iterator it = script_mappings.find(script_fn);
if(it!= script_mappings.end())
{
tar_file = it->second.tar_file;
pipe_file = it->second.pipe_file;
std::string script_fn = it->second.script_fn;
if (pipe_file != NULL)
{
script_mappings.erase(it);
}
return script_fn;
}
else
{
tar_file = false;
pipe_file = NULL;
return script_fn;
}
}
void FileServ::registerMetadataCallback( const std::string &name, const std::string& identity, IMetadataCallback* callback)
{
PipeSessions::registerMetadataCallback(name, identity, callback);
}
void FileServ::removeMetadataCallback( const std::string &name, const std::string& identity )
{
PipeSessions::removeMetadataCallback(name, identity);
}
void FileServ::registerTokenCallbackFactory( IFileServ::ITokenCallbackFactory* callback_factory )
{
IScopedLock lock(mutex);
token_callback_factory = callback_factory;
}
IFileServ::ITokenCallback* FileServ::newTokenCallback()
{
IScopedLock lock(mutex);
if(token_callback_factory==NULL)
{
return NULL;
}
return token_callback_factory->getTokenCallback();
}
void FileServ::incrShareActive(std::string sharename)
{
if (sharename.find("/") != std::string::npos)
{
sharename = getuntil("/", sharename);
}
IScopedLock lock(mutex);
++active_shares[sharename];
}
void FileServ::decrShareActive(std::string sharename)
{
if (sharename.find("/") != std::string::npos)
{
sharename = getuntil("/", sharename);
}
IScopedLock lock(mutex);
std::map::iterator it = active_shares.find(sharename);
if (it != active_shares.end())
{
--it->second;
if (it->second == 0)
{
active_shares.erase(it);
}
}
}
bool FileServ::hasActiveTransfers(const std::string& sharename, const std::string& server_token)
{
if (PipeSessions::isShareActive(sharename, server_token))
{
return true;
}
IScopedLock lock(mutex);
std::map::iterator it = active_shares.find(server_token + "|" + sharename);
return it != active_shares.end();
}
bool FileServ::registerFnRedirect(const std::string & source_fn, const std::string & target_fn)
{
IScopedLock lock(mutex);
fn_redirects[source_fn] = target_fn;
return false;
}
std::string FileServ::getRedirectedFn(const std::string & source_fn)
{
IScopedLock lock(mutex);
str_map::iterator it = fn_redirects.find(source_fn);
if (it != fn_redirects.end())
{
return it->second;
}
else
{
return source_fn;
}
}
void FileServ::callErrorCallback(std::string sharename, const std::string & filepath, int64 pos, const std::string& msg)
{
{
IScopedLock lock(mutex);
read_error_files.push_back(filepath);
}
if (sharename.find("/") != std::string::npos)
{
sharename = getuntil("/", sharename);
}
if (read_error_callback != NULL)
{
read_error_callback->onReadError(sharename, filepath, pos, msg);
}
}
bool FileServ::hasReadError(const std::string & filepath)
{
IScopedLock lock(mutex);
return std::find(read_error_files.begin(),
read_error_files.end(), filepath)!=read_error_files.end();
}
void FileServ::registerReadErrorCallback(IReadErrorCallback * cb)
{
read_error_callback = cb;
}
void FileServ::clearReadErrors()
{
IScopedLock lock(mutex);
read_error_files.clear();
}
void FileServ::clearReadErrorFile(const std::string & filepath)
{
IScopedLock lock(mutex);
std::vector::iterator it = std::find(read_error_files.begin(),
read_error_files.end(), filepath);
if (it != read_error_files.end())
{
read_error_files.erase(it);
}
}
void FileServ::setCbtHashFile(const std::string & sharename, const std::string & identity, IFileServ::CbtHashFileInfo cbt_hash_file_info)
{
IScopedLock lock(mutex);
std::map, IFileServ::CbtHashFileInfo>::iterator it = cbt_hash_files.find(std::make_pair(sharename, identity));
if (it != cbt_hash_files.end())
{
Server->destroy(it->second.cbt_hash_file);
}
cbt_hash_files[std::make_pair(sharename, identity)] = cbt_hash_file_info;
}
IFileServ::CbtHashFileInfo FileServ::getCbtHashFile(const std::string & sharename, const std::string & identity)
{
IScopedLock lock(mutex);
std::map, IFileServ::CbtHashFileInfo>::iterator it = cbt_hash_files.find(std::make_pair(sharename, identity));
if (it != cbt_hash_files.end())
{
return it->second;
}
return IFileServ::CbtHashFileInfo();
}
void FileServ::registerScriptPipeFile(const std::string & script_fn, IPipeFileExt * pipe_file)
{
IScopedLock lock(mutex);
script_mappings[script_fn] = SScriptMapping(script_fn, false, new PipeFileExt(pipe_file, script_fn));
}
void FileServ::deregisterScriptPipeFile(const std::string & script_fn)
{
IScopedLock lock(mutex);
std::map::iterator it = script_mappings.find(script_fn);
if (it != script_mappings.end())
{
script_mappings.erase(it);
}
}