urbackup_backend/urbackupclient/clientdao.cpp
2013-08-15 13:06:20 +02:00

619 lines
18 KiB
C++

/*************************************************************************
* UrBackup - Client/Server backup system
* Copyright (C) 2011 Martin Raiber
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
**************************************************************************/
#include "clientdao.h"
#include "../stringtools.h"
#include "../Interface/Server.h"
#include <memory.h>
/**
* @-SQLGenTempSetup
* @sql
* CREATE TEMPORARY TABLE filehashes_tmp (name TEXT, filesize INTEGER, modifytime INTEGER, hashdata BLOB)
*/
ClientDAO::ClientDAO(IDatabase *pDB)
{
db=pDB;
prepareQueries();
prepareQueriesGen();
}
void ClientDAO::prepareQueries(void)
{
q_get_files=db->Prepare("SELECT data,num FROM files WHERE name=?", false);
q_add_files=db->Prepare("INSERT INTO files_tmp (name, num, data) VALUES (?,?,?)", false);
q_get_dirs=db->Prepare("SELECT name, path, id FROM backupdirs", false);
q_remove_all=db->Prepare("DELETE FROM files", false);
q_get_changed_dirs=db->Prepare("SELECT id, name FROM mdirs UNION SELECT id, name FROM mdirs_backup", false);
q_remove_changed_dirs=db->Prepare("DELETE FROM mdirs", false);
q_modify_files=db->Prepare("UPDATE files SET data=?, num=? WHERE name=?", false);
q_has_files=db->Prepare("SELECT count(*) AS num FROM files WHERE name=?", false);
q_insert_shadowcopy=db->Prepare("INSERT INTO shadowcopies (vssid, ssetid, target, path, tname, orig_target, filesrv, vol, starttime, refs, starttoken) VALUES (?, ?, ?, ?, ?, ?, ?, ?, CURRENT_TIMESTAMP, ?, ?)", false);
q_get_shadowcopies=db->Prepare("SELECT id, vssid, ssetid, target, path, tname, orig_target, filesrv, vol, (strftime('%s','now') - strftime('%s', starttime)) AS passedtime, refs, starttoken FROM shadowcopies", false);
q_remove_shadowcopies=db->Prepare("DELETE FROM shadowcopies WHERE id=?", false);
q_save_changed_dirs=db->Prepare("INSERT OR REPLACE INTO mdirs_backup SELECT id,name FROM mdirs", false);
q_delete_saved_changed_dirs=db->Prepare("DELETE FROM mdirs_backup", false);
q_copy_from_tmp_files=db->Prepare("INSERT INTO files (num, data, name) SELECT num, data, name FROM files_tmp", false);
q_delete_tmp_files=db->Prepare("DELETE FROM files_tmp", false);
q_has_changed_gap=db->Prepare("SELECT name FROM mdirs WHERE name GLOB '##-GAP-##*'", false);
q_get_del_dirs=db->Prepare("SELECT name FROM del_dirs UNION SELECT name FROM del_dirs_backup", false);
q_del_del_dirs=db->Prepare("DELETE FROM del_dirs", false);
q_copy_del_dirs=db->Prepare("INSERT INTO del_dirs_backup SELECT name FROM del_dirs", false);
q_del_del_dirs_copy=db->Prepare("DELETE FROM del_dirs_backup", false);
q_remove_del_dir=db->Prepare("DELETE FROM files WHERE name GLOB ?", false);
q_get_shadowcopy_refcount=db->Prepare("SELECT refs FROM shadowcopies WHERE id=?", false);
q_set_shadowcopy_refcount=db->Prepare("UPDATE shadowcopies SET refs=? WHERE id=?", false);
q_save_changed_files=db->Prepare("INSERT OR REPLACE INTO mfiles_backup SELECT dir_id,name FROM mfiles", false);
q_remove_changed_files=db->Prepare("DELETE FROM mfiles", false);
q_delete_saved_changed_files=db->Prepare("DELETE FROM mfiles_backup", false);
q_has_changed_file=db->Prepare("SELECT dir_id FROM mfiles_backup WHERE dir_id=? AND name=? UNION SELECT dir_id FROM mfiles WHERE dir_id=? AND name=?", false);
q_get_changed_files=db->Prepare("SELECT name FROM mfiles_backup WHERE dir_id=? UNION SELECT name FROM mfiles WHERE dir_id=?", false);
q_get_pattern=db->Prepare("SELECT tvalue FROM misc WHERE tkey=?", false);
q_insert_pattern=db->Prepare("INSERT INTO misc (tkey, tvalue) VALUES (?, ?)", false);
q_update_pattern=db->Prepare("UPDATE misc SET tvalue=? WHERE tkey=?", false);
q_get_file_hash=db->Prepare("SELECT hashdata, filesize, modifytime FROM filehashes WHERE name=?", false);
}
void ClientDAO::destroyQueries(void)
{
db->destroyQuery(q_get_files);
db->destroyQuery(q_add_files);
db->destroyQuery(q_get_dirs);
db->destroyQuery(q_remove_all);
db->destroyQuery(q_get_changed_dirs);
db->destroyQuery(q_remove_changed_dirs);
db->destroyQuery(q_modify_files);
db->destroyQuery(q_has_files);
db->destroyQuery(q_insert_shadowcopy);
db->destroyQuery(q_get_shadowcopies);
db->destroyQuery(q_remove_shadowcopies);
db->destroyQuery(q_save_changed_dirs);
db->destroyQuery(q_delete_saved_changed_dirs);
db->destroyQuery(q_copy_from_tmp_files);
db->destroyQuery(q_delete_tmp_files);
db->destroyQuery(q_has_changed_gap);
db->destroyQuery(q_get_del_dirs);
db->destroyQuery(q_del_del_dirs);
db->destroyQuery(q_copy_del_dirs);
db->destroyQuery(q_del_del_dirs_copy);
db->destroyQuery(q_remove_del_dir);
db->destroyQuery(q_get_shadowcopy_refcount);
db->destroyQuery(q_set_shadowcopy_refcount);
db->destroyQuery(q_save_changed_files);
db->destroyQuery(q_remove_changed_files);
db->destroyQuery(q_delete_saved_changed_files);
db->destroyQuery(q_has_changed_file);
db->destroyQuery(q_get_changed_files);
db->destroyQuery(q_get_pattern);
db->destroyQuery(q_insert_pattern);
db->destroyQuery(q_update_pattern);
db->destroyQuery(q_get_file_hash);
}
//@-SQLGenSetup
void ClientDAO::prepareQueriesGen(void)
{
q_modifyFileHash=db->Prepare("UPDATE filehashes SET hashdata=?, filesize=?, modifytime=? WHERE name=?", false);
q_addFileHash=db->Prepare("INSERT INTO filehashes_tmp (name, filesize, modifytime, hashdata) VALUES (?, ?, ?, ?)", false);
q_copyFromTmpFileHashes=db->Prepare("INSERT INTO filehashes (name, filesize, modifytime, hashdata) SELECT name, filesize, modifytime, hashdata FROM filehashes_tmp", false);
q_deleteTmpFileHashes=db->Prepare("DELETE FROM filehashes_tmp", false);
}
//@-SQLGenDestruction
void ClientDAO::destroyQueriesGen(void)
{
db->destroyQuery(q_modifyFileHash);
db->destroyQuery(q_addFileHash);
db->destroyQuery(q_copyFromTmpFileHashes);
db->destroyQuery(q_deleteTmpFileHashes);
}
void ClientDAO::restartQueries(void)
{
destroyQueries();
destroyQueriesGen();
prepareQueries();
prepareQueriesGen();
}
bool ClientDAO::getFiles(std::wstring path, std::vector<SFile> &data)
{
q_get_files->Bind(path);
db_results res=q_get_files->Read();
q_get_files->Reset();
if(res.size()==0)
return false;
std::wstring &qdata=res[0][L"data"];
if(qdata.empty())
return true;
int num=watoi(res[0][L"num"]);
char *ptr=(char*)&qdata[0];
while(ptr-(char*)&qdata[0]<num)
{
SFile f;
unsigned short ss;
memcpy(&ss, ptr, sizeof(unsigned short));
ptr+=sizeof(unsigned short);
std::string tmp;
tmp.resize(ss);
memcpy(&tmp[0], ptr, ss);
f.name=Server->ConvertToUnicode(tmp);
ptr+=ss;
memcpy(&f.size, ptr, sizeof(int64));
ptr+=sizeof(int64);
memcpy(&f.last_modified, ptr, sizeof(int64));
ptr+=sizeof(int64);
char isdir=*ptr;
++ptr;
if(isdir==0)
f.isdir=false;
else
f.isdir=true;
data.push_back(f);
}
return true;
}
char * constructData(const std::vector<SFile> &data, size_t &datasize)
{
datasize=0;
std::vector<std::string> utf;
for(size_t i=0;i<data.size();++i)
{
std::string us=Server->ConvertToUTF8(data[i].name);
datasize+=us.size();
datasize+=sizeof(unsigned short);
datasize+=sizeof(int64)*2;
++datasize;
utf.push_back(us);
}
char *buffer=new char[datasize];
char *ptr=buffer;
for(size_t i=0;i<data.size();++i)
{
unsigned short ss=(unsigned short)utf[i].size();
memcpy(ptr, (char*)&ss, sizeof(unsigned short));
ptr+=sizeof(unsigned short);
memcpy(ptr, &utf[i][0], ss);
ptr+=ss;
memcpy(ptr, (char*)&data[i].size, sizeof(int64));
ptr+=sizeof(int64);
memcpy(ptr, (char*)&data[i].last_modified, sizeof(int64));
ptr+=sizeof(int64);
char isdir=1;
if(!data[i].isdir)
isdir=0;
*ptr=isdir;
ptr+=sizeof(char);
}
return buffer;
}
void ClientDAO::addFiles(std::wstring path, const std::vector<SFile> &data)
{
size_t ds;
char *buffer=constructData(data, ds);
q_add_files->Bind(path);
q_add_files->Bind(ds);
q_add_files->Bind(buffer, (_u32)ds);
q_add_files->Write();
q_add_files->Reset();
delete []buffer;
}
void ClientDAO::modifyFiles(std::wstring path, const std::vector<SFile> &data)
{
size_t ds;
char *buffer=constructData(data, ds);
q_modify_files->Bind(buffer, (_u32)ds);
q_modify_files->Bind(ds);
q_modify_files->Bind(path);
q_modify_files->Write();
q_modify_files->Reset();
delete []buffer;
}
bool ClientDAO::hasFiles(std::wstring path)
{
q_has_files->Bind(path);
db_results res=q_has_files->Read();
q_has_files->Reset();
if(res.size()>0)
return res[0][L"num"]==L"1";
else
return false;
}
std::vector<SBackupDir> ClientDAO::getBackupDirs(void)
{
db_results res=q_get_dirs->Read();
q_get_dirs->Reset();
std::vector<SBackupDir> ret;
for(size_t i=0;i<res.size();++i)
{
SBackupDir dir;
dir.id=watoi(res[i][L"id"]);
dir.tname=res[i][L"name"];
dir.path=res[i][L"path"];
if(dir.tname!=L"*")
ret.push_back(dir);
}
return ret;
}
void ClientDAO::removeAllFiles(void)
{
q_remove_all->Write();
}
std::vector<SMDir> ClientDAO::getChangedDirs(bool del)
{
std::vector<SMDir> ret;
db->BeginTransaction();
if(del)
{
q_save_changed_dirs->Write();
q_save_changed_dirs->Reset();
q_remove_changed_dirs->Write();
q_remove_changed_dirs->Reset();
}
db_results res=q_get_changed_dirs->Read();
q_get_changed_dirs->Reset();
db->EndTransaction();
for(size_t i=0;i<res.size();++i)
{
ret.push_back(SMDir(watoi64(res[i][L"id"]), res[i][L"name"] ) );
}
return ret;
}
void ClientDAO::moveChangedFiles(bool del)
{
if(del)
{
db->BeginTransaction();
q_save_changed_files->Write();
q_save_changed_files->Reset();
q_remove_changed_files->Write();
q_remove_changed_files->Reset();
db->EndTransaction();
}
}
std::vector<SShadowCopy> ClientDAO::getShadowcopies(void)
{
db_results res=q_get_shadowcopies->Read();
q_get_shadowcopies->Reset();
std::vector<SShadowCopy> ret;
for(size_t i=0;i<res.size();++i)
{
db_single_result &r=res[i];
SShadowCopy sc;
sc.id=watoi(r[L"id"]);
memcpy(&sc.vssid, r[L"vssid"].c_str(), sizeof(GUID) );
memcpy(&sc.ssetid, r[L"ssetid"].c_str(), sizeof(GUID) );
sc.target=r[L"target"];
sc.path=r[L"path"];
sc.tname=r[L"tname"];
sc.orig_target=r[L"orig_target"];
sc.filesrv=r[L"filesrv"]==L"0"?false:true;
sc.vol=r[L"vol"];
sc.passedtime=watoi(r[L"passedtime"]);
sc.refs=watoi(r[L"refs"]);
sc.starttoken=r[L"starttoken"];
ret.push_back(sc);
}
return ret;
}
int ClientDAO::addShadowcopy(const SShadowCopy &sc)
{
q_insert_shadowcopy->Bind((char*)&sc.vssid, sizeof(GUID) );
q_insert_shadowcopy->Bind((char*)&sc.ssetid, sizeof(GUID) );
q_insert_shadowcopy->Bind(sc.target);
q_insert_shadowcopy->Bind(sc.path);
q_insert_shadowcopy->Bind(sc.tname);
q_insert_shadowcopy->Bind(sc.orig_target);
q_insert_shadowcopy->Bind(sc.filesrv?1:0);
q_insert_shadowcopy->Bind(sc.vol);
q_insert_shadowcopy->Bind(sc.refs);
q_insert_shadowcopy->Bind(sc.starttoken);
q_insert_shadowcopy->Write();
q_insert_shadowcopy->Reset();
return (int)db->getLastInsertID();
}
int ClientDAO::modShadowcopyRefCount(int id, int m)
{
q_get_shadowcopy_refcount->Bind(id);
db_results res=q_get_shadowcopy_refcount->Read();
q_get_shadowcopy_refcount->Reset();
if(!res.empty())
{
int refs=watoi(res[0][L"refs"]);
refs+=m;
q_set_shadowcopy_refcount->Bind(refs);
q_set_shadowcopy_refcount->Bind(id);
q_set_shadowcopy_refcount->Write();
q_set_shadowcopy_refcount->Reset();
return refs;
}
return -1;
}
void ClientDAO::deleteShadowcopy(int id)
{
q_remove_shadowcopies->Bind(id);
q_remove_shadowcopies->Write();
q_remove_shadowcopies->Reset();
}
void ClientDAO::deleteSavedChangedDirs(void)
{
q_delete_saved_changed_dirs->Write();
q_delete_saved_changed_dirs->Reset();
}
void ClientDAO::deleteSavedChangedFiles(void)
{
q_delete_saved_changed_files->Write();
q_delete_saved_changed_files->Reset();
}
void ClientDAO::copyFromTmpFiles(void)
{
q_copy_from_tmp_files->Write();
q_copy_from_tmp_files->Reset();
q_delete_tmp_files->Write();
q_delete_tmp_files->Reset();
}
bool ClientDAO::hasChangedGap(void)
{
db_results res=q_has_changed_gap->Read();
q_has_changed_gap->Reset();
return !res.empty();
}
void ClientDAO::deleteChangedDirs(void)
{
q_remove_changed_dirs->Write();
q_remove_changed_dirs->Reset();
}
std::vector<std::wstring> ClientDAO::getGapDirs(void)
{
db_results res=q_has_changed_gap->Read();
q_has_changed_gap->Reset();
std::vector<std::wstring> ret;
for(size_t i=0;i<res.size();++i)
{
std::wstring gap=res[i][L"name"];
gap.erase(0,9);
ret.push_back(gap);
}
return ret;
}
std::vector<std::wstring> ClientDAO::getDelDirs(bool del)
{
std::vector<std::wstring> ret;
db->BeginTransaction();
if(del)
{
q_copy_del_dirs->Write();
q_copy_del_dirs->Reset();
q_del_del_dirs->Write();
q_del_del_dirs->Reset();
}
db_results res=q_get_del_dirs->Read();
q_get_del_dirs->Reset();
db->EndTransaction();
for(size_t i=0;i<res.size();++i)
{
ret.push_back(res[i][L"name"]);
}
return ret;
}
void ClientDAO::deleteSavedDelDirs(void)
{
q_del_del_dirs_copy->Write();
q_del_del_dirs_copy->Reset();
}
void ClientDAO::removeDeletedDir(const std::wstring &dir)
{
q_remove_del_dir->Bind(dir+L"*");
q_remove_del_dir->Write();
q_remove_del_dir->Reset();
}
bool ClientDAO::hasFileChange(_i64 dir_id, std::wstring fn)
{
q_has_changed_file->Bind(dir_id);
q_has_changed_file->Bind(fn);
q_has_changed_file->Bind(dir_id);
q_has_changed_file->Bind(fn);
db_results res=q_has_changed_file->Read();
q_has_changed_file->Reset();
return !res.empty();
}
std::vector<std::wstring> ClientDAO::getChangedFiles(_i64 dir_id)
{
q_get_changed_files->Bind(dir_id);
q_get_changed_files->Bind(dir_id);
db_results res=q_get_changed_files->Read();
q_get_changed_files->Reset();
std::vector<std::wstring> ret;
ret.resize(res.size());
for(size_t i=0;i<res.size();++i)
{
ret[i]=res[i][L"name"];
}
return ret;
}
const std::string exclude_pattern_key="exclude_pattern";
std::wstring ClientDAO::getOldExcludePattern(void)
{
return getMiscValue(exclude_pattern_key);
}
void ClientDAO::updateOldExcludePattern(const std::wstring &pattern)
{
updateMiscValue(exclude_pattern_key, pattern);
}
const std::string include_pattern_key="include_pattern";
std::wstring ClientDAO::getOldIncludePattern(void)
{
return getMiscValue(include_pattern_key);
}
void ClientDAO::updateOldIncludePattern(const std::wstring &pattern)
{
updateMiscValue(include_pattern_key, pattern);
}
std::wstring ClientDAO::getMiscValue(const std::string& key)
{
q_get_pattern->Bind(key);
db_results res=q_get_pattern->Read();
q_get_pattern->Reset();
if(!res.empty())
{
return res[0][L"tvalue"];
}
else
{
return L"";
}
}
void ClientDAO::updateMiscValue(const std::string& key, const std::wstring& value)
{
q_get_pattern->Bind(exclude_pattern_key);
db_results res=q_get_pattern->Read();
q_get_pattern->Reset();
if(!res.empty())
{
q_update_pattern->Bind(value);
q_update_pattern->Bind(key);
q_update_pattern->Write();
q_update_pattern->Reset();
}
else
{
q_insert_pattern->Bind(key);
q_insert_pattern->Bind(value);
q_insert_pattern->Write();
q_insert_pattern->Reset();
}
}
bool ClientDAO::getFileHash(const std::wstring& path, _i64& filesize, _i64& modifytime, std::string& hash)
{
q_get_file_hash->Bind(path);
db_results res=q_get_file_hash->Read();
q_get_file_hash->Reset();
if(!res.empty())
{
std::wstring &hdata=res[0][L"hashdata"];
hash.resize(hdata.size()*sizeof(wchar_t));
memcpy(const_cast<char*>(hash.c_str()), hdata.c_str(), hdata.size()*sizeof(wchar_t));
filesize=watoi64(res[0][L"filesize"]);
modifytime=watoi64(res[0][L"modifytime"]);
return true;
}
return false;
}
/**
* @-SQLGenAccess
* @func void ClientDAO::modifyFileHash
* @sql
* UPDATE filehashes SET hashdata=:hash(blob), filesize=:filesize(int64), modifytime=:modifytime(int64) WHERE name=:path(string)
*/
void ClientDAO::modifyFileHash(const std::string& hash, int64 filesize, int64 modifytime, const std::wstring& path)
{
q_modifyFileHash->Bind(hash.c_str(), (_u32)hash.size());
q_modifyFileHash->Bind(filesize);
q_modifyFileHash->Bind(modifytime);
q_modifyFileHash->Bind(path);
q_modifyFileHash->Reset();
}
/**
* @-SQLGenAccess
* @func void ClientDAO::addFileHash
* @sql
* INSERT INTO filehashes_tmp (name, filesize, modifytime, hashdata) VALUES (:name(string), :filesize(int64), :modifytime(int64), :hashdata(blob))
*/
void ClientDAO::addFileHash(const std::wstring& name, int64 filesize, int64 modifytime, const std::string& hashdata)
{
q_addFileHash->Bind(name);
q_addFileHash->Bind(filesize);
q_addFileHash->Bind(modifytime);
q_addFileHash->Bind(hashdata.c_str(), (_u32)hashdata.size());
q_addFileHash->Write();
q_addFileHash->Reset();
}
/**
* @-SQLGenAccess
* @func void ClientDAO::copyFromTmpFileHashes
* @sql
* INSERT INTO filehashes (name, filesize, modifytime, hashdata) SELECT name, filesize, modifytime, hashdata FROM filehashes_tmp
*/
void ClientDAO::copyFromTmpFileHashes(void)
{
q_copyFromTmpFileHashes->Write();
}
/**
* @-SQLGenAccess
* @func void ClientDAO::deleteTmpFileHashes
* @sql
* DELETE FROM filehashes_tmp
*/
void ClientDAO::deleteTmpFileHashes(void)
{
q_deleteTmpFileHashes->Write();
}