mirror of
https://github.com/uroni/urbackup_backend.git
synced 2025-10-26 11:36:50 +00:00
501 lines
13 KiB
C++
501 lines
13 KiB
C++
/*************************************************************************
|
|
* 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 <http://www.gnu.org/licenses/>.
|
|
**************************************************************************/
|
|
|
|
#include "../Interface/Database.h"
|
|
#include "../Interface/Server.h"
|
|
#include "../Interface/File.h"
|
|
#include "../Interface/DatabaseCursor.h"
|
|
#include "database.h"
|
|
#include "../stringtools.h"
|
|
#include <iostream>
|
|
#include <fstream>
|
|
#include "../urbackupcommon/sha2/sha2.h"
|
|
#include "../urbackupcommon/os_functions.h"
|
|
#include <memory.h>
|
|
#include <memory>
|
|
#include "dao/ServerFilesDao.h"
|
|
#include "FileIndex.h"
|
|
#include "create_files_index.h"
|
|
#include "server_hash.h"
|
|
#include "serverinterface/helper.h"
|
|
#include "server.h"
|
|
#include "../urbackupcommon/TreeHash.h"
|
|
|
|
const _u32 c_read_blocksize=4096;
|
|
const size_t draw_segments=30;
|
|
const size_t c_speed_size=15;
|
|
const size_t c_max_l_length=80;
|
|
|
|
void draw_progress(std::string curr_fn, _i64 curr_verified, _i64 verify_size)
|
|
{
|
|
static _i64 last_progress_bytes=0;
|
|
static int64 last_time=0;
|
|
static size_t max_line_length=0;
|
|
|
|
int64 passed_time=Server->getTimeMS()-last_time;
|
|
if(passed_time>1000)
|
|
{
|
|
_i64 new_bytes=curr_verified-last_progress_bytes;
|
|
|
|
float pc_done=(float)curr_verified/(float)verify_size;
|
|
|
|
size_t segments=(size_t)(pc_done*draw_segments);
|
|
|
|
std::string toc="\r[";
|
|
for(size_t i=0;i<draw_segments;++i)
|
|
{
|
|
if(i<segments)
|
|
{
|
|
toc+="=";
|
|
}
|
|
else if(i==segments)
|
|
{
|
|
toc+=">";
|
|
}
|
|
else
|
|
{
|
|
toc+=" ";
|
|
}
|
|
}
|
|
std::string speed_str=PrettyPrintSpeed((size_t)((new_bytes*1000)/passed_time));
|
|
while(speed_str.size()<c_speed_size)
|
|
speed_str+=" ";
|
|
std::string pcdone=convert((int)(pc_done*100.f));
|
|
if(pcdone.size()==1)
|
|
pcdone=" "+pcdone;
|
|
|
|
toc+="] "+pcdone+"% "+speed_str+" "+(curr_fn);
|
|
|
|
if(toc.size()>=c_max_l_length)
|
|
toc=toc.substr(0, c_max_l_length);
|
|
|
|
if(toc.size()>max_line_length)
|
|
max_line_length=toc.size();
|
|
|
|
while(toc.size()<max_line_length)
|
|
toc+=" ";
|
|
|
|
std::cout << toc;
|
|
std::cout.flush();
|
|
|
|
last_progress_bytes=curr_verified;
|
|
last_time=Server->getTimeMS();
|
|
}
|
|
}
|
|
|
|
class VerifyProgressCallback : public BackupServerPrepareHash::IHashProgressCallback
|
|
{
|
|
public:
|
|
VerifyProgressCallback(std::string curr_fn, _i64& curr_verified, _i64 verify_size)
|
|
: curr_fn(curr_fn), curr_verified(curr_verified), verify_size(verify_size),
|
|
curr_last(0)
|
|
{
|
|
|
|
}
|
|
|
|
virtual void hash_progress(int64 curr)
|
|
{
|
|
int64 add = curr - curr_last;
|
|
curr_last = curr;
|
|
curr_verified += add;
|
|
draw_progress(curr_fn, curr_verified, verify_size);
|
|
}
|
|
|
|
private:
|
|
std::string curr_fn;
|
|
_i64& curr_verified;
|
|
_i64 verify_size;
|
|
_i64 curr_last;
|
|
};
|
|
|
|
bool verify_file(db_single_result &res, _i64 &curr_verified, _i64 verify_size, bool& missing, const std::string& backuppath)
|
|
{
|
|
std::string fp=res["fullpath"];
|
|
std::unique_ptr<IFsFile> f(Server->openFile(os_file_prefix(fp), MODE_READ));
|
|
if( f.get()==NULL )
|
|
{
|
|
std::cout << std::endl;
|
|
Server->Log("Error opening file \""+fp+"\"", LL_ERROR);
|
|
missing = true;
|
|
return false;
|
|
}
|
|
|
|
bool in_backup_scripts = false;
|
|
if (!backuppath.empty())
|
|
{
|
|
size_t backuppath_pos = fp.find(backuppath);
|
|
if (backuppath_pos != std::string::npos)
|
|
{
|
|
if (fp.size() > backuppath_pos + backuppath.size())
|
|
{
|
|
std::string next_fp = fp.substr(backuppath_pos + backuppath.size() + 1);
|
|
std::string next_fp_folder = getuntil(os_file_sep(), next_fp);
|
|
|
|
if (next_fp_folder == "urbackup_backup_scripts")
|
|
{
|
|
in_backup_scripts = true;
|
|
}
|
|
else if (next_fp == "windows_components_config" + os_file_sep() + "backupcom.xml")
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if(watoi64(res["filesize"])!=f->Size())
|
|
{
|
|
std::cout << std::endl;
|
|
Server->Log("Filesize of \""+fp+"\" is wrong", LL_ERROR);
|
|
return false;
|
|
}
|
|
|
|
std::string f_name=ExtractFileName(fp);
|
|
|
|
VerifyProgressCallback progress_callback(f_name, curr_verified, verify_size);
|
|
FsExtentIterator extent_iterator(f.get(), 512*1024);
|
|
|
|
std::string calc_dig;
|
|
if (BackupServer::useTreeHashing() && !in_backup_scripts)
|
|
{
|
|
TreeHash treehash(NULL);
|
|
if (BackupServerPrepareHash::hash_sha(f.get(), &extent_iterator, true, treehash, &progress_callback))
|
|
{
|
|
calc_dig = treehash.finalize();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
HashSha512 shahash;
|
|
if (BackupServerPrepareHash::hash_sha(f.get(), &extent_iterator, !in_backup_scripts, shahash, &progress_callback))
|
|
{
|
|
calc_dig = shahash.finalize();
|
|
}
|
|
}
|
|
|
|
if(calc_dig.empty())
|
|
{
|
|
std::cout << std::endl;
|
|
Server->Log("Could not read all bytes of file \""+fp+"\"", LL_ERROR);
|
|
return false;
|
|
}
|
|
|
|
if(res["shahash"]!=calc_dig)
|
|
{
|
|
std::cout << std::endl;
|
|
Server->Log("Hash of \""+fp+"\" is wrong", LL_ERROR);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool verify_hashes(std::string arg)
|
|
{
|
|
IDatabase *db=Server->getDatabase(Server->getThreadID(), URBACKUPDB_SERVER);
|
|
|
|
std::string working_dir=(Server->getServerWorkingDir());
|
|
std::string v_output_fn=working_dir+os_file_sep()+"urbackup"+os_file_sep()+"verification_result.txt";
|
|
std::fstream v_failure;
|
|
v_failure.open(v_output_fn.c_str(), std::ios::out|std::ios::binary);
|
|
if( !v_failure.is_open() )
|
|
Server->Log("Could not open \""+v_output_fn+"\" for writing", LL_ERROR);
|
|
else
|
|
Server->Log("Writing verification results to \""+v_output_fn+"\"", LL_INFO);
|
|
|
|
BackupServer::setupUseTreeHashing();
|
|
|
|
std::string clientname;
|
|
std::string backupname;
|
|
|
|
if(arg!="true")
|
|
{
|
|
if(arg.find("/")==std::string::npos)
|
|
{
|
|
clientname=arg;
|
|
}
|
|
else
|
|
{
|
|
clientname=getuntil("/", arg);
|
|
backupname=getafter("/", arg);
|
|
|
|
}
|
|
}
|
|
|
|
bool delete_failed = Server->getServerParameter("delete_verify_failed")=="true";
|
|
|
|
int cid=0;
|
|
int backupid=0;
|
|
std::string filter;
|
|
|
|
IDatabase *files_db = Server->getDatabase(Server->getThreadID(), URBACKUPDB_SERVER_FILES);
|
|
|
|
if(!clientname.empty())
|
|
{
|
|
if(clientname!="*")
|
|
{
|
|
IQuery *q=db->Prepare("SELECT id FROM clients WHERE name=?");
|
|
q->Bind(clientname);
|
|
db_results res=q->Read();
|
|
if(!res.empty())
|
|
{
|
|
cid=watoi(res[0]["id"]);
|
|
}
|
|
else
|
|
{
|
|
Server->Log("Client \""+clientname+"\" not found", LL_ERROR);
|
|
return false;
|
|
}
|
|
|
|
filter="clientid="+convert(cid);
|
|
}
|
|
else
|
|
{
|
|
filter+="1=1";
|
|
}
|
|
|
|
|
|
if(!backupname.empty())
|
|
{
|
|
std::string backupid_filter;
|
|
std::string temp_create_query;
|
|
if(backupname=="last")
|
|
{
|
|
if(clientname!="*")
|
|
{
|
|
IQuery* q=db->Prepare("SELECT id,path FROM backups WHERE clientid=? AND complete=1 ORDER BY backuptime DESC LIMIT 1");
|
|
q->Bind(cid);
|
|
db_results res=q->Read();
|
|
if(!res.empty())
|
|
{
|
|
backupid_filter="= "+res[0]["id"];
|
|
Server->Log("Last backup: "+res[0]["path"], LL_INFO);
|
|
}
|
|
else
|
|
{
|
|
Server->Log("Last backup not found", LL_ERROR);
|
|
return false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
backupid_filter = " IN (SELECT id FROM backups)";
|
|
temp_create_query = "SELECT id FROM backups b WHERE NOT EXISTS (SELECT id FROM backups c WHERE complete=1 AND c.backuptime > b.backuptime AND b.clientid=c.clientid)";
|
|
}
|
|
|
|
}
|
|
else if(backupname=="*")
|
|
{
|
|
backupid_filter=" IN (SELECT id FROM backups)";
|
|
temp_create_query = "SELECT id FROM backups WHERE clientid=" + convert(cid);
|
|
}
|
|
else
|
|
{
|
|
IQuery* q=db->Prepare("SELECT id FROM backups WHERE path=? AND clientid=?");
|
|
q->Bind(backupname);
|
|
q->Bind(cid);
|
|
db_results res=q->Read();
|
|
if(!res.empty())
|
|
{
|
|
backupid_filter="= "+res[0]["id"];
|
|
}
|
|
else
|
|
{
|
|
Server->Log("Backup \""+backupname+"\" not found", LL_ERROR);
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if(backupname!="*" || clientname!="*")
|
|
{
|
|
filter+=" AND backupid "+backupid_filter;
|
|
}
|
|
|
|
if (!temp_create_query.empty())
|
|
{
|
|
files_db->Write("CREATE TEMPORARY TABLE backups (id INTEGER PRIMARY KEY)");
|
|
|
|
IQuery* q_insert = files_db->Prepare("INSERT INTO backups (id) VALUES (?)", false);
|
|
IQuery* q_backup_ids = db->Prepare(temp_create_query, false);
|
|
IDatabaseCursor* cur = q_backup_ids->Cursor();
|
|
|
|
db_single_result res;
|
|
while (cur->next(res))
|
|
{
|
|
q_insert->Bind(res["id"]);
|
|
q_insert->Write();
|
|
q_insert->Reset();
|
|
}
|
|
|
|
db->destroyQuery(q_backup_ids);
|
|
files_db->destroyQuery(q_insert);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
filter = "1=1";
|
|
}
|
|
|
|
|
|
std::cout << "Calculating filesize..." << std::endl;
|
|
IQuery *q_num_files = files_db->Prepare("SELECT SUM(filesize) AS c FROM files WHERE filesize>0 AND "+filter);
|
|
db_results res=q_num_files->Read();
|
|
if(res.empty())
|
|
{
|
|
Server->Log("Error during filesize calculation.", LL_ERROR);
|
|
return false;
|
|
}
|
|
|
|
_i64 verify_size=watoi64(res[0]["c"]);
|
|
_i64 curr_verified=0;
|
|
|
|
std::cout << "To be verified: " << PrettyPrintBytes(verify_size) << " of files" << std::endl;
|
|
|
|
_i64 crowid=0;
|
|
|
|
IQuery *q_get_files = files_db->Prepare("SELECT id, fullpath, shahash, filesize, backupid FROM files WHERE "+filter, false);
|
|
IQuery* q_get_backuppath = db->Prepare("SELECT path FROM backups WHERE id=?", false);
|
|
|
|
bool is_okay=true;
|
|
|
|
IDatabaseCursor* cursor = q_get_files->Cursor();
|
|
|
|
std::vector<int64> todelete;
|
|
std::vector<int64> missing_files;
|
|
std::map<int, std::string> backuppaths;
|
|
|
|
db_single_result res_single;
|
|
while(cursor->next(res_single))
|
|
{
|
|
int backupid = watoi(res_single["backupid"]);
|
|
std::string backuppath;
|
|
std::map<int, std::string>::iterator it_backuppath = backuppaths.find(backupid);
|
|
if (it_backuppath == backuppaths.end())
|
|
{
|
|
q_get_backuppath->Bind(backupid);
|
|
db_results res_backuppath = q_get_backuppath->Read();
|
|
q_get_backuppath->Reset();
|
|
if (!res_backuppath.empty())
|
|
{
|
|
backuppath = res_backuppath[0]["path"];
|
|
backuppaths.insert(std::make_pair(backupid, backuppath));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
backuppath = it_backuppath->second;
|
|
}
|
|
|
|
bool is_missing=false;
|
|
if(! verify_file( res_single, curr_verified, verify_size, is_missing, backuppath) )
|
|
{
|
|
if(!is_missing)
|
|
{
|
|
v_failure << "Verification of \"" << (res_single["fullpath"]) << "\" failed\r\n";
|
|
is_okay=false;
|
|
|
|
if(delete_failed)
|
|
{
|
|
todelete.push_back(watoi64(res_single["id"]));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
missing_files.push_back(watoi64(res_single["id"]));
|
|
}
|
|
}
|
|
}
|
|
|
|
std::cout << std::endl;
|
|
|
|
if(v_failure.is_open() && is_okay)
|
|
{
|
|
v_failure.close();
|
|
Server->deleteFile(v_output_fn);
|
|
}
|
|
|
|
files_db->destroyQuery(q_get_files);
|
|
db->destroyQuery(q_get_backuppath);
|
|
|
|
IQuery* q_get_file = files_db->Prepare("SELECT id, fullpath, shahash, filesize, backupid FROM files WHERE id=?");
|
|
|
|
if (missing_files.size() > 0)
|
|
{
|
|
std::cout << missing_files.size() << " could not be opened during verification. Checking now if they have been deleted from the database..." << std::endl;
|
|
|
|
for (size_t i = 0; i < missing_files.size(); ++i)
|
|
{
|
|
q_get_file->Bind(missing_files[i]);
|
|
db_results res = q_get_file->Read();
|
|
q_get_file->Reset();
|
|
|
|
if (!res.empty())
|
|
{
|
|
bool is_missing = false;
|
|
db_single_result& res_single = res[0];
|
|
if (!verify_file(res_single, curr_verified, verify_size, is_missing, backuppaths[watoi(res_single["backupid"])]))
|
|
{
|
|
v_failure << "Verification of file \"" << (res_single["fullpath"]) << "\" failed (during rechecking previously missing files)\r\n";
|
|
is_okay = false;
|
|
|
|
if (delete_failed)
|
|
{
|
|
todelete.push_back(watoi64(res_single["id"]));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if(delete_failed)
|
|
{
|
|
std::cout << "Deleting " << todelete.size() << " file entries with failed verification from database..." << std::endl;
|
|
|
|
SStartupStatus status;
|
|
if(!create_files_index(status))
|
|
{
|
|
std::cout << "Error opening file index -1" << std::endl;
|
|
}
|
|
else
|
|
{
|
|
files_db = Server->getDatabase(Server->getThreadID(), URBACKUPDB_SERVER_FILES);
|
|
ServerFilesDao backupdao(files_db);
|
|
std::unique_ptr<FileIndex> fileindex(create_lmdb_files_index());
|
|
|
|
if(fileindex.get()==NULL)
|
|
{
|
|
std::cout << "Error opening file index -2" << std::endl;
|
|
}
|
|
else
|
|
{
|
|
|
|
for(size_t i=0;i<todelete.size();++i)
|
|
{
|
|
BackupServerHash::deleteFileSQL(backupdao, *fileindex, todelete[i]);
|
|
}
|
|
|
|
std::cout << "done." << std::endl;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
return is_okay;
|
|
} |