mirror of
https://github.com/uroni/urbackup_backend.git
synced 2025-10-26 11:36:50 +00:00
531 lines
11 KiB
C++
531 lines
11 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 "vld.h"
|
|
#define DEF_SERVER
|
|
#include "Server.h"
|
|
#include "AcceptThread.h"
|
|
#include "SessionMgr.h"
|
|
#include "LoadbalancerClient.h"
|
|
#include "libs.h"
|
|
#include "sqlite/sqlite3.h"
|
|
#include "stringtools.h"
|
|
#include <iostream>
|
|
#include <vector>
|
|
#include <map>
|
|
#include <stdlib.h>
|
|
#include <time.h>
|
|
|
|
#ifdef _WIN32
|
|
#include <conio.h>
|
|
#endif
|
|
#ifdef AS_SERVICE
|
|
# include "win_service/nt_service.h"
|
|
using namespace nt;
|
|
#endif
|
|
|
|
#ifndef _WIN32
|
|
# include <sys/types.h>
|
|
# include <pwd.h>
|
|
#endif
|
|
|
|
CServer *Server=NULL;
|
|
|
|
using namespace std;
|
|
|
|
bool run=true;
|
|
bool no_server=false;
|
|
|
|
void init_mutex_selthread(void);
|
|
void destroy_mutex_selthread(void);
|
|
|
|
#ifndef _WIN32
|
|
void termination_handler(int signum)
|
|
{
|
|
run=false;
|
|
Server->Log("Shutting down (Signal "+nconvert(signum)+")", LL_WARNING);
|
|
}
|
|
#endif
|
|
|
|
#ifdef AS_SERVICE
|
|
char **srv_argv;
|
|
int srv_argc;
|
|
#endif
|
|
|
|
void DisplayHelp(void)
|
|
{
|
|
cout << "CServer - Compiled C++ Server" << endl;
|
|
cout << "CServer [--port 34255] [--workers 3] [--plugin x] [--loadbalancer 192.168.0.1] [--lb-weight 1] [--lb-port 2304]" << endl;
|
|
}
|
|
|
|
CAcceptThread *c_at=NULL;
|
|
|
|
#ifndef AS_SERVICE
|
|
int main(int argc, char *argv[])
|
|
{
|
|
#else
|
|
int my_init_fcn_t(int argc, char *argv[])
|
|
{
|
|
#endif
|
|
Server=new CServer;
|
|
Server->setup();
|
|
|
|
#ifdef _WIN32
|
|
int rc;
|
|
|
|
WSADATA wsadata;
|
|
rc = WSAStartup(MAKEWORD(2,2), &wsadata);
|
|
if(rc == SOCKET_ERROR)
|
|
{
|
|
Server->Log("Error starting Winsock 2.2", LL_ERROR);
|
|
return 23;
|
|
}
|
|
#endif
|
|
|
|
srand((unsigned int)time(0));
|
|
|
|
//Parse Parameters
|
|
|
|
int port=34255;
|
|
int workers=5;
|
|
|
|
std::vector<std::string> plugins;
|
|
|
|
std::string loadbalancer;
|
|
int loadbalancer_weight=1;
|
|
int loadbalancer_port=2305;
|
|
|
|
if( argc==2 && (std::string)argv[1]=="--version" )
|
|
{
|
|
cout << "CServer version 0.1.0.0" << endl;
|
|
return 1;
|
|
}
|
|
|
|
if( argc==2 && ( (std::string)argv[1]=="--help" || (std::string)argv[1]=="-h") )
|
|
{
|
|
DisplayHelp();
|
|
return 6;
|
|
}
|
|
|
|
std::map<std::string, std::string> srv_params;
|
|
|
|
std::string loglevel;
|
|
std::string logfile;
|
|
std::string workingdir;
|
|
bool daemon=false;
|
|
std::string daemon_user;
|
|
std::string pidfile;
|
|
|
|
for(int i=1;i<argc;++i)
|
|
{
|
|
std::string carg=argv[i];
|
|
std::string narg;
|
|
if( i+1<argc )
|
|
narg=argv[i+1];
|
|
|
|
if( carg=="--daemon" || carg=="-d" )
|
|
{
|
|
daemon=true;
|
|
}
|
|
else if( carg=="--user" || carg=="-u" )
|
|
{
|
|
daemon_user=narg;
|
|
++i;
|
|
}
|
|
else if( carg=="--workingdir" )
|
|
{
|
|
workingdir=narg;
|
|
++i;
|
|
}
|
|
else if( carg=="--port" || carg=="-p" )
|
|
{
|
|
port=atoi( narg.c_str() );
|
|
++i;
|
|
}
|
|
else if( carg=="--workers" || carg=="-w" )
|
|
{
|
|
workers=atoi( narg.c_str() );
|
|
++i;
|
|
}
|
|
else if( carg=="--plugin" || carg=="-p" || carg=="--add" || carg=="-a")
|
|
{
|
|
plugins.push_back( narg );
|
|
++i;
|
|
}
|
|
else if( carg=="--loadbalancer" || carg=="-lb" )
|
|
{
|
|
loadbalancer=narg;
|
|
++i;
|
|
}
|
|
else if( carg=="--lb-weight" || carg=="-lbw" )
|
|
{
|
|
loadbalancer_weight=atoi( narg.c_str() );
|
|
++i;
|
|
}
|
|
else if( carg=="--lb-port" || carg=="-lbp" )
|
|
{
|
|
loadbalancer_port=atoi( narg.c_str() );
|
|
++i;
|
|
}
|
|
else if( carg=="--no-server" || carg=="-n" )
|
|
{
|
|
no_server=true;
|
|
}
|
|
else if (carg=="--loglevel" || carg=="-ll" )
|
|
{
|
|
loglevel=strlower(narg);
|
|
++i;
|
|
}
|
|
else if (carg=="--logfile" || carg=="-lf" )
|
|
{
|
|
logfile=narg;
|
|
++i;
|
|
}
|
|
else if( carg=="--pidfile" )
|
|
{
|
|
pidfile=narg;
|
|
++i;
|
|
}
|
|
else
|
|
{
|
|
if( carg.size()>1 && carg[0]=='-' )
|
|
{
|
|
std::string p;
|
|
if( carg[1]=='-' && carg.size()>2 )
|
|
p=carg.substr(2, carg.size()-2);
|
|
else
|
|
p=carg.substr(1, carg.size()-1);
|
|
|
|
std::string v="true";
|
|
|
|
if( narg.size()>0 && narg[0]!='-' )
|
|
{
|
|
v=narg;
|
|
++i;
|
|
}
|
|
else
|
|
{
|
|
size_t g=p.find_last_of("=");
|
|
if(g!=std::string::npos)
|
|
{
|
|
v=p.substr(g+1);
|
|
p=p.substr(0,g);
|
|
}
|
|
}
|
|
|
|
|
|
srv_params[p]=v;
|
|
}
|
|
}
|
|
}
|
|
|
|
Server->setServerParameters(srv_params);
|
|
|
|
if(workingdir.empty())
|
|
{
|
|
#ifdef _WIN32
|
|
#ifndef AS_SERVICE
|
|
{
|
|
wchar_t buf[MAX_PATH];
|
|
GetCurrentDirectory(MAX_PATH, buf);
|
|
Server->setServerWorkingDir(buf);
|
|
}
|
|
#else
|
|
{
|
|
wchar_t buf[MAX_PATH+1];
|
|
GetModuleFileNameW(NULL, buf, MAX_PATH);
|
|
Server->setServerWorkingDir(ExtractFilePath(buf));
|
|
SetCurrentDirectory(ExtractFilePath(buf).c_str());
|
|
}
|
|
#endif
|
|
#else
|
|
Server->setServerWorkingDir(Server->ConvertToUnicode(ExtractFilePath(argv[0])));
|
|
#endif
|
|
}
|
|
else
|
|
{
|
|
Server->setServerWorkingDir(Server->ConvertToUnicode(workingdir));
|
|
}
|
|
|
|
|
|
#ifndef _WIN32
|
|
if(daemon)
|
|
{
|
|
if( fork()==0 )
|
|
{
|
|
setsid();
|
|
for (int i=getdtablesize();i>=0;--i) close(i);
|
|
int i=open("/dev/null",O_RDWR);
|
|
dup(i);
|
|
dup(i);
|
|
}
|
|
else
|
|
exit(0);
|
|
|
|
chdir(Server->ConvertToUTF8(Server->getServerWorkingDir()).c_str());
|
|
|
|
if(pidfile.empty())
|
|
{
|
|
pidfile="/var/run/urbackup_srv.pid";
|
|
}
|
|
|
|
std::fstream pf;
|
|
pf.open(pidfile.c_str(), std::ios::out|std::ios::binary);
|
|
if(pf.is_open())
|
|
{
|
|
pf << getpid();
|
|
pf.close();
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
|
|
|
|
if(!logfile.empty())
|
|
{
|
|
Server->setLogFile(logfile, daemon_user);
|
|
}
|
|
|
|
|
|
if(!loglevel.empty())
|
|
{
|
|
if(loglevel=="debug")
|
|
Server->setLogLevel(LL_DEBUG);
|
|
else if(loglevel=="warn")
|
|
Server->setLogLevel(LL_WARNING);
|
|
else if(loglevel=="info")
|
|
Server->setLogLevel(LL_INFO);
|
|
else if(loglevel=="error")
|
|
Server->setLogLevel(LL_ERROR);
|
|
}
|
|
|
|
#ifndef _WIN32
|
|
if( !daemon_user.empty() && (getuid()==0 || geteuid()==0) )
|
|
{
|
|
Server->Log("Changing user...", LL_DEBUG);
|
|
char buf[1000];
|
|
passwd pwbuf;
|
|
passwd *pw;
|
|
int rc=getpwnam_r(daemon_user.c_str(), &pwbuf, buf, 1000, &pw);
|
|
if(pw!=NULL)
|
|
{
|
|
Server->Log("done.");
|
|
setgid(pw->pw_gid);
|
|
setuid(pw->pw_uid);
|
|
}
|
|
else
|
|
{
|
|
Server->Log("Unable to change user, probably because process uid is not root", LL_ERROR);
|
|
}
|
|
}
|
|
#endif
|
|
for( size_t i=0;i<plugins.size();++i)
|
|
{
|
|
if( !Server->LoadDLL(plugins[i]) )
|
|
{
|
|
Server->Log("Loading "+(std::string)plugins[i]+" failed", LL_ERROR);
|
|
}
|
|
}
|
|
|
|
CLoadbalancerClient *lbs=NULL;
|
|
if( loadbalancer!="" )
|
|
{
|
|
lbs=new CLoadbalancerClient( loadbalancer, loadbalancer_port, loadbalancer_weight, port);
|
|
|
|
Server->createThread(lbs);
|
|
}
|
|
|
|
#ifndef _WIN32
|
|
if (signal (SIGINT, termination_handler) == SIG_IGN)
|
|
signal (SIGINT, SIG_IGN);
|
|
if(!daemon)
|
|
{
|
|
if (signal (SIGHUP, termination_handler) == SIG_IGN)
|
|
signal (SIGHUP, SIG_IGN);
|
|
}
|
|
if (signal (SIGTERM, termination_handler) == SIG_IGN)
|
|
signal (SIGTERM, SIG_IGN);
|
|
#endif
|
|
|
|
if( sqlite3_threadsafe()==0 )
|
|
{
|
|
Server->Log("SQLite3 wasn't compiled with the SQLITE_THREADSAFE. Exiting.", LL_ERROR);
|
|
return 43;
|
|
}
|
|
|
|
//sqlite3_enable_shared_cache(1);
|
|
|
|
((CSessionMgr*)Server->getSessionMgr())->startTimeoutSessionThread();
|
|
|
|
|
|
if(no_server==false )
|
|
{
|
|
init_mutex_selthread();
|
|
c_at=new CAcceptThread(workers, port);
|
|
if(c_at->has_error())
|
|
{
|
|
Server->Log("Error while starting listening to ports. Stopping server.", LL_ERROR);
|
|
run=false;
|
|
}
|
|
#ifndef AS_SERVICE
|
|
while(run==true)
|
|
{
|
|
(*c_at)(true);
|
|
#ifdef _WIN32
|
|
if( _kbhit()!=0 )
|
|
{
|
|
break;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
Server->Log("Exited Loop");
|
|
|
|
Server->Log("Deleting at..");
|
|
delete c_at;
|
|
|
|
destroy_mutex_selthread();
|
|
}
|
|
else
|
|
{
|
|
while(run==true)
|
|
{
|
|
Server->wait(1000);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
#ifndef AS_SERVICE
|
|
Server->Log("Deleting lbs...");
|
|
delete lbs;
|
|
Server->Log("Shutting down plugins...");
|
|
Server->ShutdownPlugins();
|
|
|
|
Server->Log("Deleting server...");
|
|
delete Server;
|
|
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
#ifdef AS_SERVICE
|
|
void WINAPI my_service_main(DWORD argc, char_* argv[])
|
|
{
|
|
if( c_at!=NULL )
|
|
{
|
|
(*c_at)(true);
|
|
}
|
|
else if(no_server==false)
|
|
{
|
|
Sleep(500);
|
|
//throw std::exception("Service stopped");
|
|
}
|
|
else
|
|
{
|
|
Sleep(500);
|
|
}
|
|
}
|
|
void my_shutdown_fcn(void)
|
|
{
|
|
/*if(c_at !=NULL )
|
|
{
|
|
delete c_at;
|
|
destroy_mutex_selthread();
|
|
c_at=NULL;
|
|
}
|
|
if(Server!=NULL)
|
|
{
|
|
delete Server;
|
|
Server=NULL;
|
|
}*/
|
|
}
|
|
void my_stop_fcn(void)
|
|
{
|
|
/*if(c_at !=NULL )
|
|
{
|
|
CAcceptThread *tmp=c_at;
|
|
c_at=NULL;
|
|
delete tmp;
|
|
delete Server;
|
|
}
|
|
if(Server!=NULL)
|
|
{
|
|
CServer *srv=Server;
|
|
Server=NULL;
|
|
delete srv;
|
|
}*/
|
|
nt_service& service = nt_service::instance(L"CompiledServer");
|
|
service.stop(0);
|
|
}
|
|
|
|
void my_init_fcn(void)
|
|
{
|
|
my_init_fcn_t(srv_argc, srv_argv);
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
SetCurrentDirectoryA(ExtractFilePath(argv[0]).c_str() );
|
|
std::string args=getFile("args.txt");
|
|
int lc=linecount(args);
|
|
srv_argv=new char*[lc+1];
|
|
srv_argv[0]=new char[strlen(argv[0])+1];
|
|
strcpy_s(srv_argv[0], strlen(argv[0])+1, argv[0]);
|
|
for(int i=0;i<lc;++i)
|
|
{
|
|
std::string l=getline(i, args);
|
|
std::cout << l << std::endl;
|
|
srv_argv[i+1]=new char[l.size()+1];
|
|
memcpy(srv_argv[i+1], &l[0], l.size());
|
|
srv_argv[i+1][l.size()]=0;
|
|
}
|
|
|
|
srv_argc=lc+1;
|
|
|
|
if( argc>1 && (std::string)argv[1]=="pgo" )
|
|
{
|
|
my_init_fcn();
|
|
|
|
while(true)
|
|
{
|
|
my_service_main(0,NULL);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// creates an access point to the instance of the service framework
|
|
nt_service& service = nt_service::instance(L"CompiledServer");
|
|
|
|
// register "my_service_main" to be executed as the service main method
|
|
service.register_service_main( my_service_main );
|
|
|
|
// register "my_init_fcn" as initialization fcn
|
|
service.register_init_function( my_init_fcn );
|
|
|
|
// config the service to accepts stop controls. Do nothing when it happens
|
|
//service.accept_control( SERVICE_ACCEPT_STOP );
|
|
service.register_control_handler( SERVICE_CONTROL_STOP, my_stop_fcn );
|
|
|
|
// config the service to accepts shutdown controls and do something when receive it
|
|
service.register_control_handler( SERVICE_CONTROL_SHUTDOWN, my_shutdown_fcn );
|
|
|
|
service.start();
|
|
}
|
|
}
|
|
#endif |