/************************************************************************* * 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 "restore_client.h" #include "../Interface/Thread.h" #include "../fileservplugin/IFileServ.h" #include "../Interface/File.h" #include "../Interface/Server.h" #include "server_settings.h" #include "ClientMain.h" #include #include "../urbackupcommon/file_metadata.h" #include "serverinterface/backups.h" #include "../urbackupcommon/filelist_utils.h" #include "../common/data.h" #include "database.h" #include "dao/ServerBackupDao.h" #include "dao/ServerCleanupDao.h" extern IFileServ* fileserv; namespace { const int64 win32_meta_magic = little_endian(0x320FAB3D119DCB4AULL); const int64 unix_meta_magic = little_endian(0xFE4378A3467647F0ULL); const _u32 ID_METADATA_V1_WIN = 1<<0 | 1<<3; const _u32 ID_METADATA_V1_UNIX = 1<<2 | 1<<3; class MetadataCallback : public IFileServ::IMetadataCallback { public: MetadataCallback(const std::string& basedir, const std::vector < std::pair >& map_paths) : basedir(basedir), map_paths(map_paths) { } virtual IFile* getMetadata( const std::string& path, std::string& orig_path, int64& offset, int64& length, _u32& version ) { if(path.empty()) return NULL; std::string metadata_path = basedir; std::vector path_segments; TokenizeMail(path.substr(1), path_segments, "/"); bool isdir = path[0]=='d'; for(size_t i=1;i metadata_file(Server->openFile(os_file_prefix(metadata_path), MODE_READ)); if(metadata_file.get()==NULL) { return NULL; } FileMetadata metadata; if(!read_metadata(metadata_file.get(), metadata)) { return NULL; } orig_path = metadata.orig_path; for (size_t j = 0; j < map_paths.size(); ++j) { if (next(orig_path, 0, map_paths[j].first)) { orig_path.replace(orig_path.begin(), orig_path.begin() + map_paths[j].first.size(), map_paths[j].second); break; } } offset = os_metadata_offset(metadata_file.get()); length = metadata_file->Size() - offset; int64 metadata_magic_and_size[2]; if(!metadata_file->Seek(offset)) { return NULL; } if(metadata_file->Read(reinterpret_cast(&metadata_magic_and_size), sizeof(metadata_magic_and_size))!=sizeof(metadata_magic_and_size)) { return NULL; } if(metadata_magic_and_size[1]==win32_meta_magic) { version=ID_METADATA_V1_WIN; } else if(metadata_magic_and_size[1]==unix_meta_magic) { version=ID_METADATA_V1_UNIX; } else { version=0; } if(!metadata_file->Seek(offset)) { return NULL; } return metadata_file.release(); } private: std::string basedir; std::vector < std::pair > map_paths; }; class ClientDownloadThread : public IThread { public: ClientDownloadThread(const std::string& curr_clientname, int curr_clientid, int restore_clientid, IFile* filelist_f, const std::string& foldername, const std::string& hashfoldername, const std::string& filter, bool token_authentication, const std::vector &backup_tokens, const std::vector &tokens, bool skip_special_root, const std::string& folder_log_name, int64 restore_id, size_t status_id, logid_t log_id, const std::string& restore_token, const std::string& identity, const std::vector >& map_paths, bool clean_other, bool ignore_other_fs) : curr_clientname(curr_clientname), curr_clientid(curr_clientid), restore_clientid(restore_clientid), filelist_f(filelist_f), foldername(foldername), hashfoldername(hashfoldername), token_authentication(token_authentication), backup_tokens(backup_tokens), tokens(tokens), skip_special_root(skip_special_root), folder_log_name(folder_log_name), restore_token(restore_token), identity(identity), restore_id(restore_id), status_id(status_id), log_id(log_id), single_file(false), map_paths(map_paths), clean_other(clean_other), ignore_other_fs(ignore_other_fs) { TokenizeMail(filter, filter_fns, "/"); if (filter_fns.size()==1) { single_file = true; } } void operator()() { IDatabase* db = Server->getDatabase(Server->getThreadID(), URBACKUPDB_SERVER); ServerBackupDao backup_dao(db); if(!createFilelist(foldername, hashfoldername, 0, skip_special_root)) { ServerStatus::stopProcess(curr_clientname, status_id); int errors=0; int warnings=0; int infos=0; std::string logdata=ServerLogger::getLogdata(log_id, errors, warnings, infos); backup_dao.saveBackupLog(restore_clientid, errors, warnings, infos, 0, 0, 0, 1); backup_dao.saveBackupLogData(db->getLastInsertID(), logdata); backup_dao.setRestoreDone(0, restore_id); return; } fileserv->addIdentity(identity); fileserv->shareDir("clientdl_filelist", filelist_f->getFilename(), identity); ClientMain::addShareToCleanup(curr_clientid, SShareCleanup("clientdl_filelist", identity, true, false)); delete filelist_f; MetadataCallback* callback = new MetadataCallback(hashfoldername, map_paths); fileserv->shareDir("clientdl", foldername, identity); fileserv->shareDir("urbackup", "/tmp/mkmergsdfklrzrehmklregmfdkgfdgwretklödf", identity); ClientMain::addShareToCleanup(curr_clientid, SShareCleanup("clientdl", identity, false, true)); ClientMain::addShareToCleanup(curr_clientid, SShareCleanup("urbackup", identity, false, false)); fileserv->registerMetadataCallback("clientdl", identity, callback); CWData data; data.addBuffer("RESTORE", 7); data.addString(identity); data.addInt64(restore_id); data.addUInt64(status_id); data.addInt64(log_id.first); data.addString(restore_token); data.addChar(single_file ? 1 : 0); data.addChar(clean_other ? 1 : 0); data.addChar(ignore_other_fs ? 1 : 0); std::string msg(data.getDataPtr(), data.getDataPtr()+data.getDataSize()); ServerStatus::sendToCommPipe(curr_clientname, msg); } bool writeFile(const std::string& data) { std::string towrite = (data); return filelist_f->Write(towrite)==towrite.size(); } bool createFilelist(const std::string& foldername, const std::string& hashfoldername, size_t depth, bool skip_special) { bool has_error=false; const std::vector files = getFiles(os_file_prefix(foldername), &has_error); if(has_error) { ServerLogger::Log(log_id, "Cannot read files from folder "+foldername+". Cannot start restore.", LL_ERROR); return false; } bool ret=true; for(size_t i=0;i filter_fns; bool token_authentication; std::vector backup_tokens; std::vector tokens; bool skip_special_root; std::string folder_log_name; int64 restore_id; size_t status_id; logid_t log_id; std::string restore_token; std::string identity; bool single_file; std::vector < std::pair > map_paths; bool clean_other; bool ignore_other_fs; }; } bool create_clientdl_thread(const std::string& curr_clientname, int curr_clientid, int restore_clientid, std::string foldername, std::string hashfoldername, const std::string& filter, bool token_authentication, const std::vector &backup_tokens, const std::vector &tokens, bool skip_hashes, const std::string& folder_log_name, int64& restore_id, size_t& status_id, logid_t& log_id, const std::string& restore_token, const std::vector >& map_paths, bool clean_other, bool ignore_other_fs) { IFile* filelist_f = Server->openTemporaryFile(); if(filelist_f==NULL) { return false; } if(!foldername.empty() && foldername[foldername.size()-1]==os_file_sep()[0]) { foldername = foldername.substr(0, foldername.size()-1); } if(!hashfoldername.empty() && hashfoldername[hashfoldername.size()-1]==os_file_sep()[0]) { hashfoldername = hashfoldername.substr(0, hashfoldername.size()-1); } IDatabase* db = Server->getDatabase(Server->getThreadID(), URBACKUPDB_SERVER); ServerBackupDao backup_dao(db); std::string full_log_name; if (filter.find("/") != std::string::npos) { full_log_name = folder_log_name + "[" + filter + "]"; } else { full_log_name = folder_log_name + filter; } std::string identity = ServerSettings::generateRandomAuthKey(25); backup_dao.addRestore(restore_clientid, full_log_name, identity, 0, std::string()); restore_id = db->getLastInsertID(); log_id = ServerLogger::getLogId(restore_clientid); status_id = ServerStatus::startProcess(curr_clientname, sa_restore_file, full_log_name, log_id, false); Server->getThreadPool()->execute(new ClientDownloadThread(curr_clientname, curr_clientid, restore_clientid, filelist_f, foldername, hashfoldername, filter, token_authentication, backup_tokens, tokens, skip_hashes, folder_log_name, restore_id, status_id, log_id, restore_token, identity, map_paths, clean_other, ignore_other_fs), "frestore preparation"); return true; } bool create_clientdl_thread( int backupid, const std::string& curr_clientname, int curr_clientid, int64& restore_id, size_t& status_id, logid_t& log_id, const std::string& restore_token, const std::vector >& map_paths, bool clean_other, bool ignore_other_fs) { IDatabase* db = Server->getDatabase(Server->getThreadID(), URBACKUPDB_SERVER); ServerBackupDao backup_dao(db); ServerCleanupDao cleanup_dao(db); ServerBackupDao::SFileBackupInfo file_backup_info = backup_dao.getFileBackupInfo(backupid); if(!file_backup_info.exists) { Server->Log("Could not get file backup info for backupid "+convert(backupid)); return false; } ServerCleanupDao::CondString client_name = cleanup_dao.getClientName(file_backup_info.clientid); if(!client_name.exists) { Server->Log("Could not get client name for clientid "+convert(file_backup_info.clientid)); return false; } ServerBackupDao::CondString backupfolder = backup_dao.getSetting(0, "backupfolder"); if(!backupfolder.exists) { Server->Log("Could not get backup storage folder"); return false; } std::string curr_path=backupfolder.value+os_file_sep()+client_name.value+os_file_sep()+file_backup_info.path; std::string curr_metadata_path=backupfolder.value+os_file_sep()+client_name.value+os_file_sep()+file_backup_info.path+os_file_sep()+".hashes"; std::vector backup_tokens; std::vector tokens; return create_clientdl_thread(curr_clientname, curr_clientid, file_backup_info.clientid, curr_path, curr_metadata_path, std::string(), false, backup_tokens, tokens, true, "", restore_id, status_id, log_id, restore_token, map_paths, clean_other, ignore_other_fs); }