urbackup_backend/urbackupserver/apps/cleanup_cmd.cpp
Martin b70cc08e48 * Handle changes of the backupfolder correctly
* Lazy load database queries
* Bug fixes for symlink pooling
2014-04-01 20:47:29 +02:00

229 lines
5.7 KiB
C++

#include "app.h"
#include "../server_settings.h"
#include "../../urbackupcommon/os_functions.h"
#include "../../stringtools.h"
#include "../server_cleanup.h"
#include "../server.h"
int64 cleanup_amount(std::string cleanup_pc, IDatabase *db)
{
ServerSettings settings(db);
int64 total_space=os_total_space(settings.getSettings()->backupfolder);
if(total_space==-1)
{
Server->Log("Error getting free space", LL_ERROR);
return -1;
}
strupper(&cleanup_pc);
std::wstring wcleanup_pc=widen(cleanup_pc);
int64 cleanup_bytes=0;
if(cleanup_pc.find("%")!=std::string::npos)
{
double pc=atof(getuntil("%", cleanup_pc).c_str());
Server->Log("Cleaning up "+nconvert(pc)+" percent", LL_INFO);
cleanup_bytes=(int64)((pc/100)*total_space+0.5);
}
else if(cleanup_pc.find("K")!=std::string::npos)
{
cleanup_bytes=watoi64(getuntil(L"K", wcleanup_pc))*1024;
}
else if(cleanup_pc.find("M")!=std::string::npos)
{
cleanup_bytes=watoi64(getuntil(L"M", wcleanup_pc))*1024*1024;
}
else if(cleanup_pc.find("G")!=std::string::npos)
{
cleanup_bytes=watoi64(getuntil(L"G", wcleanup_pc))*1024*1024*1024;
}
else if(cleanup_pc.find("T")!=std::string::npos)
{
cleanup_bytes=watoi64(getuntil(L"T", wcleanup_pc))*1024*1024*1024*1024;
}
else
{
cleanup_bytes=watoi64(wcleanup_pc);
}
if(cleanup_bytes>total_space)
{
cleanup_bytes=total_space;
}
return cleanup_bytes;
}
int cleanup_cmd(void)
{
Server->Log("Shutting down all database instances...", LL_INFO);
Server->destroyAllDatabases();
Server->Log("Opening urbackup server database...", LL_INFO);
bool use_bdb;
open_server_database(use_bdb, true);
open_settings_database(use_bdb);
IDatabase *db=Server->getDatabase(Server->getThreadID(), URBACKUPDB_SERVER);
if(db==NULL)
{
Server->Log("Could not open database", LL_ERROR);
return 1;
}
BackupServer::testSnapshotAvailability(db);
Server->Log("Transitioning urbackup server database to different journaling mode...", LL_INFO);
db->Write("PRAGMA journal_mode = DELETE");
std::string cleanup_pc=Server->getServerParameter("cleanup_amount");
if(cleanup_pc=="true" || cleanup_pc.empty())
{
cleanup_pc="10%";
}
int64 cleanup_bytes=cleanup_amount(cleanup_pc, db);
if(cleanup_bytes<0)
{
return 3;
}
Server->Log("Cleaning up "+PrettyPrintBytes(cleanup_bytes)+" on backup storage", LL_INFO);
{
ServerSettings settings(db);
size_t cache_size=settings.getSettings()->update_stats_cachesize;
Server->Log("Database cache size is "+PrettyPrintBytes(cache_size*1024), LL_INFO);
}
Server->Log("Starting cleanup...", LL_INFO);
Server->Log("Freeing database connections...", LL_INFO);
Server->destroyAllDatabases();
bool b = ServerCleanupThread::cleanupSpace(cleanup_bytes, true);
if(!b)
{
Server->Log("Cleanup failed. Could not remove backups. Please lower the minimal number of backups", LL_ERROR);
return 2;
}
Server->Log("Cleanup successfull.", LL_INFO);
return 0;
}
int defrag_database(void)
{
Server->Log("Shutting down all database instances...", LL_INFO);
Server->destroyAllDatabases();
Server->Log("Opening urbackup server database...", LL_INFO);
bool use_bdb;
open_server_database(use_bdb, true);
IDatabase *db=Server->getDatabase(Server->getThreadID(), URBACKUPDB_SERVER);
if(db==NULL)
{
Server->Log("Could not open database", LL_ERROR);
return 1;
}
Server->Log("Transitioning urbackup server database to different journaling mode...", LL_INFO);
db->Write("PRAGMA journal_mode = DELETE");
Server->Log("Rebuilding Database...", LL_INFO);
db->Write("PRAGMA page_size = 4096");
db->Write("VACUUM");
Server->Log("Rebuilding Database successfull.", LL_INFO);
Server->Log("Deleting file entry cache, if present...", LL_INFO);
delete_file_caches();
Server->Log("Done.");
return 0;
}
int remove_unknown(void)
{
Server->Log("Going to remove all unknown files and directories in the urbackup storage directory. Waiting 20 seconds...", LL_INFO);
Server->wait(20000);
Server->setServerParameter("cleanup_amount", "0%");
if(cleanup_cmd()!=0)
{
Server->Log("Error cleaning up.", LL_ERROR);
return 1;
}
ServerCleanupThread::removeUnknown();
Server->Log("Successfully removed all unknown files in backup directory.", LL_INFO);
return 0;
}
int cleanup_database(void)
{
Server->Log("Going to clean up unnessecary information in database. Waiting 20 seconds...", LL_INFO);
Server->wait(20000);
Server->Log("Shutting down all database instances...", LL_INFO);
Server->destroyAllDatabases();
Server->Log("Opening urbackup server database...", LL_INFO);
bool use_bdb;
open_server_database(use_bdb, true);
open_settings_database(use_bdb);
IDatabase *db=Server->getDatabase(Server->getThreadID(), URBACKUPDB_SERVER);
if(db==NULL)
{
Server->Log("Could not open database", LL_ERROR);
return 1;
}
Server->Log("Transitioning urbackup server database to different journaling mode...", LL_INFO);
db->Write("PRAGMA journal_mode = DELETE");
db_results res=db->Read("SELECT * FROM sqlite_master WHERE type='table'");
for(size_t i=0;i<res.size();++i)
{
db_results rc=db->Read("SELECT count(*) AS c FROM "+wnarrow(res[i][L"name"]));
if(!rc.empty())
{
Server->Log(L"Table "+res[i][L"name"]+L" has "+rc[0][L"c"]+L" rows", LL_INFO);
}
}
Server->Log("Cleaning up information...", LL_INFO);
db_results rc=db->Read("SELECT count(*) AS c FROM del_stats");
if(!rc.empty())
{
if(watoi64(rc[0][L"c"])>10000000)
{
db->Write("DELETE FROM del_stats");
}
}
Server->Log("Cleaning up database...", LL_INFO);
db->Write("VACUUM");
return 0;
}