urbackup_backend/urbackupclient/clientdao.cpp
2017-02-13 23:32:41 +01:00

969 lines
26 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 "clientdao.h"
#include "../stringtools.h"
#include "../Interface/Server.h"
#include <memory.h>
const int ClientDAO::c_is_group = 0;
const int ClientDAO::c_is_user = 1;
const int ClientDAO::c_is_system_user = 2;
ClientDAO::ClientDAO(IDatabase *pDB)
{
db=pDB;
prepareQueries();
}
ClientDAO::~ClientDAO()
{
destroyQueries();
}
void ClientDAO::prepareQueries()
{
q_get_files=db->Prepare("SELECT data, num, generation FROM files WHERE name=? AND tgroup=?", false);
q_add_files=db->Prepare("INSERT OR REPLACE INTO files (name, tgroup, num, data) VALUES (?,?,?,?)", false);
q_get_dirs=db->Prepare("SELECT name, path, id, optional, tgroup, symlinked, server_default, reset_keep FROM backupdirs", false);
q_remove_all=db->Prepare("DELETE FROM files", false);
q_get_changed_dirs=db->Prepare("SELECT id, name FROM mdirs WHERE name GLOB ? UNION SELECT id, name FROM mdirs_backup WHERE name GLOB ?", false);
q_modify_files=db->Prepare("UPDATE files SET data=?, num=?, generation=? WHERE name=? AND tgroup=? AND generation=?", false);
q_has_files=db->Prepare("SELECT count(*) AS num FROM files WHERE name=? AND tgroup=?", false);
q_insert_shadowcopy=db->Prepare("INSERT INTO shadowcopies (vssid, ssetid, target, path, tname, orig_target, filesrv, vol, starttime, refs, starttoken, clientsubname) 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, clientsubname 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 WHERE name GLOB ?", false);
q_delete_saved_changed_dirs=db->Prepare("DELETE FROM mdirs_backup", 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 WHERE name GLOB ? UNION SELECT name FROM del_dirs_backup WHERE name GLOB ?", false);
q_del_del_dirs=db->Prepare("DELETE FROM del_dirs WHERE name GLOB ?", false);
q_copy_del_dirs=db->Prepare("INSERT INTO del_dirs_backup SELECT name FROM del_dirs WHERE name GLOB ?", 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 ? AND tgroup=?", 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_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);
prepareQueriesGen();
}
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_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_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_get_pattern);
db->destroyQuery(q_insert_pattern);
db->destroyQuery(q_update_pattern);
destroyQueriesGen();
}
//@-SQLGenSetup
void ClientDAO::prepareQueriesGen(void)
{
q_updateShadowCopyStarttime=NULL;
q_updateFileAccessToken=NULL;
q_getFileAccessTokens=NULL;
q_getFileAccessTokenId2Alts=NULL;
q_getFileAccessTokenId=NULL;
q_removeGroupMembership=NULL;
q_updateGroupMembership=NULL;
q_getGroupMembership=NULL;
q_addBackupDir=NULL;
q_delBackupDir=NULL;
q_setResetKeep=NULL;
q_resetHardlink=NULL;
q_hasHardLink=NULL;
q_addHardlink=NULL;
q_resetAllHardlinks=NULL;
}
//@-SQLGenDestruction
void ClientDAO::destroyQueriesGen(void)
{
db->destroyQuery(q_updateShadowCopyStarttime);
db->destroyQuery(q_updateFileAccessToken);
db->destroyQuery(q_getFileAccessTokens);
db->destroyQuery(q_getFileAccessTokenId2Alts);
db->destroyQuery(q_getFileAccessTokenId);
db->destroyQuery(q_removeGroupMembership);
db->destroyQuery(q_updateGroupMembership);
db->destroyQuery(q_getGroupMembership);
db->destroyQuery(q_addBackupDir);
db->destroyQuery(q_delBackupDir);
db->destroyQuery(q_setResetKeep);
db->destroyQuery(q_resetHardlink);
db->destroyQuery(q_hasHardLink);
db->destroyQuery(q_addHardlink);
db->destroyQuery(q_resetAllHardlinks);
}
void ClientDAO::restartQueries(void)
{
destroyQueries();
prepareQueries();
}
std::string ClientDAO::escapeGlob(const std::string& glob)
{
std::string ret;
ret.reserve(glob.size());
for (size_t i = 0; i<glob.size(); ++i)
{
if (glob[i] == '?')
{
ret += "[?]";
}
else if (glob[i] == '[')
{
ret += "[[]";
}
else if (glob[i] == '*')
{
ret += "[*]";
}
else
{
ret += glob[i];
}
}
return ret;
}
bool ClientDAO::getFiles(std::string path, int tgroup, std::vector<SFileAndHash> &data, int64& generation)
{
q_get_files->Bind(path);
q_get_files->Bind(tgroup);
db_results res=q_get_files->Read();
q_get_files->Reset();
if(res.size()==0)
return false;
generation = watoi64(res[0]["generation"]);
std::string &qdata=res[0]["data"];
if(qdata.empty())
return true;
int num=watoi(res[0]["num"]);
char *ptr=(char*)&qdata[0];
while(ptr-(char*)&qdata[0]<num)
{
SFileAndHash 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=(tmp);
ptr+=ss;
memcpy(&f.size, ptr, sizeof(int64));
ptr+=sizeof(int64);
memcpy(&f.change_indicator, ptr, sizeof(uint64));
ptr+=sizeof(uint64);
char isdir=*ptr;
++ptr;
if(isdir==0)
f.isdir=false;
else
f.isdir=true;
unsigned short hashsize;
memcpy(&hashsize, ptr, sizeof(unsigned short));
ptr+=sizeof(unsigned short);
f.hash.resize(hashsize);
if(hashsize>0)
{
memcpy(&f.hash[0], ptr, hashsize);
}
ptr+=hashsize;
char issym=*ptr;
++ptr;
f.issym=issym==0?false:true;
char isspecialf=*ptr;
++ptr;
f.isspecialf= isspecialf ==0?false:true;
if(f.issym)
{
memcpy(&ss, ptr, sizeof(unsigned short));
ptr+=sizeof(unsigned short);
if(ss>0)
{
tmp.resize(ss);
memcpy(&tmp[0], ptr, ss);
f.symlink_target=(tmp);
ptr+=ss;
}
}
data.push_back(f);
}
return true;
}
char * constructData(const std::vector<SFileAndHash> &data, size_t &datasize)
{
datasize=0;
std::vector<std::string> utf;
utf.resize(data.size());
for(size_t i=0;i<data.size();++i)
{
utf[i]=(data[i].name);
datasize+=utf[i].size();
datasize+=sizeof(unsigned short);
datasize+=sizeof(int64)*2;
++datasize; //isdir
datasize+=sizeof(unsigned short);
datasize+=data[i].hash.size();
++datasize; //issym
++datasize; //isspecial
if(data[i].issym)
{
datasize+=sizeof(unsigned short);
datasize+=(data[i].symlink_target).size();
}
}
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].change_indicator, sizeof(uint64));
ptr+=sizeof(uint64);
char isdir=1;
if(!data[i].isdir)
isdir=0;
*ptr=isdir;
ptr+=sizeof(char);
unsigned short hashsize=static_cast<unsigned short>(data[i].hash.size());
memcpy(ptr, &hashsize, sizeof(hashsize));
ptr+=sizeof(hashsize);
memcpy(ptr, data[i].hash.data(), hashsize);
ptr+=hashsize;
char issym = data[i].issym?1:0;
*ptr=issym;
++ptr;
char isspecialf = data[i].isspecialf?1:0;
*ptr= isspecialf ?1:0;
++ptr;
if(data[i].issym)
{
std::string symlink_target = (data[i].symlink_target);
ss=(unsigned short)symlink_target.size();
memcpy(ptr, (char*)&ss, sizeof(unsigned short));
ptr+=sizeof(unsigned short);
if(ss>0)
{
memcpy(ptr, &symlink_target[0], ss);
}
ptr+=ss;
}
}
return buffer;
}
std::string guidToString( GUID guid )
{
return bytesToHex(reinterpret_cast<unsigned char*>(&guid), sizeof(guid));
}
GUID randomGuid()
{
GUID ret;
Server->secureRandomFill(reinterpret_cast<char*>(&ret), sizeof(ret));
return ret;
}
void ClientDAO::addFiles(std::string path, int tgroup, const std::vector<SFileAndHash> &data)
{
size_t ds;
char *buffer=constructData(data, ds);
q_add_files->Bind(path);
q_add_files->Bind(tgroup);
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::string path, int tgroup, const std::vector<SFileAndHash> &data, int64 target_generation)
{
size_t ds;
char *buffer=constructData(data, ds);
q_modify_files->Bind(buffer, (_u32)ds);
q_modify_files->Bind(ds);
q_modify_files->Bind(target_generation+1);
q_modify_files->Bind(path);
q_modify_files->Bind(tgroup);
q_modify_files->Bind(target_generation);
q_modify_files->Write();
q_modify_files->Reset();
delete []buffer;
}
bool ClientDAO::hasFiles(std::string path, int tgroup)
{
q_has_files->Bind(path);
q_has_files->Bind(tgroup);
db_results res=q_has_files->Read();
q_has_files->Reset();
if(res.size()>0)
return res[0]["num"]=="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;
std::vector<SBackupDir> sym_ret;
for(size_t i=0;i<res.size();++i)
{
SBackupDir dir;
dir.id=watoi(res[i]["id"]);
dir.tname=res[i]["name"];
dir.path=res[i]["path"];
dir.flags=watoi(res[i]["optional"]);
dir.group=watoi(res[i]["tgroup"]);
dir.server_default=(res[i]["server_default"] == "1");
dir.symlinked=(res[i]["symlinked"]=="1");
dir.symlinked_confirmed=false;
dir.reset_keep = (res[i]["reset_keep"] == "1");
if(dir.tname!="*")
{
if(dir.symlinked)
{
sym_ret.push_back(dir);
}
else
{
ret.push_back(dir);
}
}
}
ret.insert(ret.end(), sym_ret.begin(), sym_ret.end());
return ret;
}
void ClientDAO::removeAllFiles(void)
{
q_remove_all->Write();
}
std::vector<std::string> ClientDAO::getChangedDirs(const std::string& path, bool backup)
{
std::vector<std::string> ret;
std::string sep = os_file_sep();
if(path == "##-GAP-##")
{
sep = "";
}
q_get_changed_dirs->Bind(escapeGlob(path)+sep+"*");
q_get_changed_dirs->Bind(escapeGlob(path)+sep+"*");
db_results res=q_get_changed_dirs->Read();
q_get_changed_dirs->Reset();
ret.reserve(res.size());
for(size_t i=0;i<res.size();++i)
{
ret.push_back(res[i]["name"] );
}
if (backup)
{
q_save_changed_dirs->Bind(escapeGlob(path) + sep + "*");
q_save_changed_dirs->Write();
q_save_changed_dirs->Reset();
}
return ret;
}
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["id"]);
memcpy(&sc.vssid, r["vssid"].c_str(), sizeof(GUID) );
memcpy(&sc.ssetid, r["ssetid"].c_str(), sizeof(GUID) );
sc.target=r["target"];
sc.path=r["path"];
sc.tname=r["tname"];
sc.orig_target=r["orig_target"];
sc.filesrv=r["filesrv"]=="0"?false:true;
sc.vol=r["vol"];
sc.passedtime=watoi(r["passedtime"]);
sc.refs=watoi(r["refs"]);
sc.starttoken=r["starttoken"];
sc.clientsubname = r["clientsubname"];
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->Bind(sc.clientsubname);
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]["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();
}
bool ClientDAO::hasChangedGap(void)
{
db_results res=q_has_changed_gap->Read();
q_has_changed_gap->Reset();
return !res.empty();
}
std::vector<std::string> ClientDAO::getGapDirs(void)
{
db_results res=q_has_changed_gap->Read();
q_has_changed_gap->Reset();
std::vector<std::string> ret;
for(size_t i=0;i<res.size();++i)
{
std::string gap=res[i]["name"];
gap.erase(0,9);
ret.push_back(gap);
}
return ret;
}
std::vector<std::string> ClientDAO::getDelDirs(const std::string& path, bool del)
{
std::vector<std::string> ret;
if(del)
{
q_copy_del_dirs->Bind(escapeGlob(path)+os_file_sep()+"*");
q_copy_del_dirs->Write();
q_copy_del_dirs->Reset();
q_del_del_dirs->Bind(escapeGlob(path)+os_file_sep()+"*");
q_del_del_dirs->Write();
q_del_del_dirs->Reset();
}
q_get_del_dirs->Bind(escapeGlob(path)+os_file_sep()+"*");
q_get_del_dirs->Bind(escapeGlob(path)+os_file_sep()+"*");
db_results res=q_get_del_dirs->Read();
q_get_del_dirs->Reset();
for(size_t i=0;i<res.size();++i)
{
ret.push_back(res[i]["name"]);
}
return ret;
}
void ClientDAO::deleteSavedDelDirs(void)
{
q_del_del_dirs_copy->Write();
q_del_del_dirs_copy->Reset();
}
void ClientDAO::removeDeletedDir(const std::string &dir, int tgroup)
{
q_remove_del_dir->Bind(escapeGlob(dir)+"*");
q_remove_del_dir->Bind(tgroup);
q_remove_del_dir->Write();
q_remove_del_dir->Reset();
}
const std::string exclude_pattern_key="exclude_pattern";
std::string ClientDAO::getOldExcludePattern(void)
{
return getMiscValue(exclude_pattern_key);
}
void ClientDAO::updateOldExcludePattern(const std::string &pattern)
{
updateMiscValue(exclude_pattern_key, pattern);
}
const std::string include_pattern_key="include_pattern";
std::string ClientDAO::getOldIncludePattern(void)
{
return getMiscValue(include_pattern_key);
}
void ClientDAO::updateOldIncludePattern(const std::string &pattern)
{
updateMiscValue(include_pattern_key, pattern);
}
std::string 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]["tvalue"];
}
else
{
return "";
}
}
void ClientDAO::updateMiscValue(const std::string& key, const std::string& value)
{
q_get_pattern->Bind(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();
}
}
/**
* @-SQLGenAccess
* @func void ClientDAO::updateShadowCopyStarttime
* @sql
* UPDATE shadowcopies SET starttime=CURRENT_TIMESTAMP
* WHERE id=:id(int)
*/
void ClientDAO::updateShadowCopyStarttime(int id)
{
if(q_updateShadowCopyStarttime==NULL)
{
q_updateShadowCopyStarttime=db->Prepare("UPDATE shadowcopies SET starttime=CURRENT_TIMESTAMP WHERE id=?", false);
}
q_updateShadowCopyStarttime->Bind(id);
q_updateShadowCopyStarttime->Write();
q_updateShadowCopyStarttime->Reset();
}
/**
* @-SQLGenAccess
* @func void ClientDAO::updateFileAccessToken
* @sql
* INSERT OR REPLACE INTO fileaccess_tokens
* (accountname, token, is_user)
* VALUES
* (:accountname(string), :token(string), :is_user(int))
*/
void ClientDAO::updateFileAccessToken(const std::string& accountname, const std::string& token, int is_user)
{
if(q_updateFileAccessToken==NULL)
{
q_updateFileAccessToken=db->Prepare("INSERT OR REPLACE INTO fileaccess_tokens (accountname, token, is_user) VALUES (?, ?, ?)", false);
}
q_updateFileAccessToken->Bind(accountname);
q_updateFileAccessToken->Bind(token);
q_updateFileAccessToken->Bind(is_user);
q_updateFileAccessToken->Write();
q_updateFileAccessToken->Reset();
}
/**
* @-SQLGenAccess
* @func vector<SToken> ClientDAO::getFileAccessTokens
* @return int64 id, string accountname, string token, int is_user
* @sql
* SELECT id, accountname, token, is_user
* FROM fileaccess_tokens
*/
std::vector<ClientDAO::SToken> ClientDAO::getFileAccessTokens(void)
{
if(q_getFileAccessTokens==NULL)
{
q_getFileAccessTokens=db->Prepare("SELECT id, accountname, token, is_user FROM fileaccess_tokens", false);
}
db_results res=q_getFileAccessTokens->Read();
std::vector<ClientDAO::SToken> ret;
ret.resize(res.size());
for(size_t i=0;i<res.size();++i)
{
ret[i].id=watoi64(res[i]["id"]);
ret[i].accountname=res[i]["accountname"];
ret[i].token=res[i]["token"];
ret[i].is_user=watoi(res[i]["is_user"]);
}
return ret;
}
/**
* @-SQLGenAccess
* @func int64 ClientDAO::getFileAccessTokenId2Alts
* @return int64 id
* @sql
* SELECT id
* FROM fileaccess_tokens WHERE accountname = :accountname(string) AND
* (is_user = :is_user_alt1(int) OR is_user = :is_user_alt2(int))
*/
ClientDAO::CondInt64 ClientDAO::getFileAccessTokenId2Alts(const std::string& accountname, int is_user_alt1, int is_user_alt2)
{
if(q_getFileAccessTokenId2Alts==NULL)
{
q_getFileAccessTokenId2Alts=db->Prepare("SELECT id FROM fileaccess_tokens WHERE accountname = ? AND (is_user = ? OR is_user = ?)", false);
}
q_getFileAccessTokenId2Alts->Bind(accountname);
q_getFileAccessTokenId2Alts->Bind(is_user_alt1);
q_getFileAccessTokenId2Alts->Bind(is_user_alt2);
db_results res=q_getFileAccessTokenId2Alts->Read();
q_getFileAccessTokenId2Alts->Reset();
CondInt64 ret = { false, 0 };
if(!res.empty())
{
ret.exists=true;
ret.value=watoi64(res[0]["id"]);
}
return ret;
}
/**
* @-SQLGenAccess
* @func int64 ClientDAO::getFileAccessTokenId
* @return int64 id
* @sql
* SELECT id
* FROM fileaccess_tokens WHERE accountname = :accountname(string) AND
* is_user = :is_user(int)
*/
ClientDAO::CondInt64 ClientDAO::getFileAccessTokenId(const std::string& accountname, int is_user)
{
if(q_getFileAccessTokenId==NULL)
{
q_getFileAccessTokenId=db->Prepare("SELECT id FROM fileaccess_tokens WHERE accountname = ? AND is_user = ?", false);
}
q_getFileAccessTokenId->Bind(accountname);
q_getFileAccessTokenId->Bind(is_user);
db_results res=q_getFileAccessTokenId->Read();
q_getFileAccessTokenId->Reset();
CondInt64 ret = { false, 0 };
if(!res.empty())
{
ret.exists=true;
ret.value=watoi64(res[0]["id"]);
}
return ret;
}
/**
* @-SQLGenAccess
* @func void ClientDAO::removeGroupMembership
* @sql
* DELETE FROM token_group_memberships WHERE uid=:uid(int64)
*/
void ClientDAO::removeGroupMembership(int64 uid)
{
if(q_removeGroupMembership==NULL)
{
q_removeGroupMembership=db->Prepare("DELETE FROM token_group_memberships WHERE uid=?", false);
}
q_removeGroupMembership->Bind(uid);
q_removeGroupMembership->Write();
q_removeGroupMembership->Reset();
}
/**
* @-SQLGenAccess
* @func void ClientDAO::updateGroupMembership
* @sql
* INSERT OR REPLACE INTO token_group_memberships
* (uid, gid)
* VALUES
* (:uid(int64), (SELECT id FROM fileaccess_tokens WHERE accountname = :accountname(string) AND is_user=0) )
*/
void ClientDAO::updateGroupMembership(int64 uid, const std::string& accountname)
{
if(q_updateGroupMembership==NULL)
{
q_updateGroupMembership=db->Prepare("INSERT OR REPLACE INTO token_group_memberships (uid, gid) VALUES (?, (SELECT id FROM fileaccess_tokens WHERE accountname = ? AND is_user=0) )", false);
}
q_updateGroupMembership->Bind(uid);
q_updateGroupMembership->Bind(accountname);
q_updateGroupMembership->Write();
q_updateGroupMembership->Reset();
}
/**
* @-SQLGenAccess
* @func vector<int> ClientDAO::getGroupMembership
* @return int gid
* @sql
* SELECT gid
* FROM token_group_memberships
* WHERE uid = :uid(int)
*/
std::vector<int> ClientDAO::getGroupMembership(int uid)
{
if(q_getGroupMembership==NULL)
{
q_getGroupMembership=db->Prepare("SELECT gid FROM token_group_memberships WHERE uid = ?", false);
}
q_getGroupMembership->Bind(uid);
db_results res=q_getGroupMembership->Read();
q_getGroupMembership->Reset();
std::vector<int> ret;
ret.resize(res.size());
for(size_t i=0;i<res.size();++i)
{
ret[i]=watoi(res[i]["gid"]);
}
return ret;
}
/**
* @-SQLGenAccess
* @func void ClientDAO::addBackupDir
* @sql
* INSERT INTO backupdirs
* (name, path, server_default, optional, tgroup, symlinked)
* VALUES
* (:name(string), :path(string), :server_default(int), :flags(int), :tgroup(int),
* :symlinked(int) )
**/
void ClientDAO::addBackupDir(const std::string& name, const std::string& path, int server_default, int flags, int tgroup, int symlinked)
{
if(q_addBackupDir==NULL)
{
q_addBackupDir=db->Prepare("INSERT INTO backupdirs (name, path, server_default, optional, tgroup, symlinked) VALUES (?, ?, ?, ?, ?, ? )", false);
}
q_addBackupDir->Bind(name);
q_addBackupDir->Bind(path);
q_addBackupDir->Bind(server_default);
q_addBackupDir->Bind(flags);
q_addBackupDir->Bind(tgroup);
q_addBackupDir->Bind(symlinked);
q_addBackupDir->Write();
q_addBackupDir->Reset();
}
/**
* @-SQLGenAccess
* @func void ClientDAO::delBackupDir
* @sql
* DELETE FROM backupdirs WHERE id=:id(int64)
**/
void ClientDAO::delBackupDir(int64 id)
{
if(q_delBackupDir==NULL)
{
q_delBackupDir=db->Prepare("DELETE FROM backupdirs WHERE id=?", false);
}
q_delBackupDir->Bind(id);
q_delBackupDir->Write();
q_delBackupDir->Reset();
}
/**
* @-SQLGenAccess
* @func void ClientDAO::setResetKeep
* @sql
* UPDATE backupdirs SET reset_keep=:val(int) WHERE id=:id(int64)
**/
void ClientDAO::setResetKeep(int val, int64 id)
{
if(q_setResetKeep==NULL)
{
q_setResetKeep=db->Prepare("UPDATE backupdirs SET reset_keep=? WHERE id=?", false);
}
q_setResetKeep->Bind(val);
q_setResetKeep->Bind(id);
q_setResetKeep->Write();
q_setResetKeep->Reset();
}
/**
* @-SQLGenAccess
* @func void ClientDAO::resetHardlink
* @sql
* DELETE FROM hardlinks WHERE vol=:vol(string) AND frn_high=:frn_high(int64) AND frn_low=:frn_low(int64)
**/
void ClientDAO::resetHardlink(const std::string& vol, int64 frn_high, int64 frn_low)
{
if(q_resetHardlink==NULL)
{
q_resetHardlink=db->Prepare("DELETE FROM hardlinks WHERE vol=? AND frn_high=? AND frn_low=?", false);
}
q_resetHardlink->Bind(vol);
q_resetHardlink->Bind(frn_high);
q_resetHardlink->Bind(frn_low);
q_resetHardlink->Write();
q_resetHardlink->Reset();
}
/**
* @-SQLGenAccess
* @func int64 ClientDAO::hasHardLink
* @return int64 frn_low
* @sql
* SELECT frn_low FROM hardlinks WHERE vol=:vol(string) AND frn_high=:frn_high(int64) AND frn_low=:frn_low(int64) LIMIT 1
**/
ClientDAO::CondInt64 ClientDAO::hasHardLink(const std::string& vol, int64 frn_high, int64 frn_low)
{
if(q_hasHardLink==NULL)
{
q_hasHardLink=db->Prepare("SELECT frn_low FROM hardlinks WHERE vol=? AND frn_high=? AND frn_low=? LIMIT 1", false);
}
q_hasHardLink->Bind(vol);
q_hasHardLink->Bind(frn_high);
q_hasHardLink->Bind(frn_low);
db_results res=q_hasHardLink->Read();
q_hasHardLink->Reset();
CondInt64 ret = { false, 0 };
if(!res.empty())
{
ret.exists=true;
ret.value=watoi64(res[0]["frn_low"]);
}
return ret;
}
/**
* @-SQLGenAccess
* @func void ClientDAO::addHardlink
* @sql
* INSERT OR IGNORE INTO hardlinks (vol, frn_high, frn_low, parent_frn_high, parent_frn_low)
* VALUES (:vol(string), :frn_high(int64), :frn_low(int64), :parent_frn_high(int64), :parent_frn_low(int64))
**/
void ClientDAO::addHardlink(const std::string& vol, int64 frn_high, int64 frn_low, int64 parent_frn_high, int64 parent_frn_low)
{
if(q_addHardlink==NULL)
{
q_addHardlink=db->Prepare("INSERT OR IGNORE INTO hardlinks (vol, frn_high, frn_low, parent_frn_high, parent_frn_low) VALUES (?, ?, ?, ?, ?)", false);
}
q_addHardlink->Bind(vol);
q_addHardlink->Bind(frn_high);
q_addHardlink->Bind(frn_low);
q_addHardlink->Bind(parent_frn_high);
q_addHardlink->Bind(parent_frn_low);
q_addHardlink->Write();
q_addHardlink->Reset();
}
/**
* @-SQLGenAccess
* @func void ClientDAO::resetAllHardlinks
* @sql
* DELETE FROM hardlinks
**/
void ClientDAO::resetAllHardlinks(void)
{
if(q_resetAllHardlinks==NULL)
{
q_resetAllHardlinks=db->Prepare("DELETE FROM hardlinks", false);
}
q_resetAllHardlinks->Write();
}