mirror of
				https://github.com/ElvishArtisan/rivendell.git
				synced 2025-10-31 14:13:52 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			170 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			170 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // rdservice.cpp
 | |
| //
 | |
| // Rivendell Services Manager
 | |
| //
 | |
| //   (C) Copyright 2018-2019 Fred Gleason <fredg@paravelsystems.com>
 | |
| //
 | |
| //   This program is free software; you can redistribute it and/or modify
 | |
| //   it under the terms of the GNU General Public License version 2 as
 | |
| //   published by the Free Software Foundation.
 | |
| //
 | |
| //   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, write to the Free Software
 | |
| //   Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
 | |
| //
 | |
| 
 | |
| #include <signal.h>
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <syslog.h>
 | |
| #include <unistd.h>
 | |
| #include <sys/types.h>
 | |
| 
 | |
| #include <qcoreapplication.h>
 | |
| 
 | |
| #include <rd.h>
 | |
| #include <rdapplication.h>
 | |
| #include <rdcmd_switch.h>
 | |
| #include <rdconf.h>
 | |
| 
 | |
| #include "rdservice.h"
 | |
| 
 | |
| bool global_exiting=false;
 | |
| bool global_reload_dropboxes=false;
 | |
| 
 | |
| void SigHandler(int signo)
 | |
| {
 | |
|   switch(signo) {
 | |
|   case SIGTERM:
 | |
|   case SIGINT:
 | |
|     global_exiting=true;
 | |
|     break;
 | |
| 
 | |
|   case SIGUSR1:
 | |
|     global_reload_dropboxes=true;
 | |
|     break;
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| MainObject::MainObject(QObject *parent)
 | |
|   :QObject(parent)
 | |
| {
 | |
|   QString err_msg;
 | |
|   RDApplication::ErrorType err_type=RDApplication::ErrorOk;
 | |
| 
 | |
|   svc_startup_target=MainObject::TargetAll;
 | |
| 
 | |
|   //
 | |
|   // Check for prior instance
 | |
|   //
 | |
|   if(RDGetPids("rdservice").size()>1) {
 | |
|     rda->syslog(LOG_ERR,"prior instance found");
 | |
|     exit(1);
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Open the Database
 | |
|   //
 | |
|   rda=new RDApplication("rdservice","rdservice","\n\n",this);
 | |
|   if(!rda->open(&err_msg,&err_type,false)) {
 | |
|     rda->syslog(LOG_ERR,"unable to open database [%s]",
 | |
| 		(const char *)err_msg.utf8());
 | |
|     exit(2);
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Process Startup Options
 | |
|   //
 | |
|   for(unsigned i=0;i<rda->cmdSwitch()->keys();i++) {
 | |
|     for(int j=0;j<MainObject::TargetAll;j++) {
 | |
|       MainObject::StartupTarget target=(MainObject::StartupTarget)j;
 | |
|       if(rda->cmdSwitch()->key(i)==TargetCommandString(target)) {
 | |
| 	svc_startup_target=target;
 | |
| 	rda->cmdSwitch()->setProcessed(i,true);
 | |
|       }
 | |
|     }
 | |
|     if(!rda->cmdSwitch()->processed(i)) {
 | |
|       fprintf(stderr,"rdservice: unknown command-line option\n");
 | |
|       exit(4);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Exit Timer
 | |
|   //
 | |
|   ::signal(SIGINT,SigHandler);
 | |
|   ::signal(SIGTERM,SigHandler);
 | |
|   ::signal(SIGUSR1,SigHandler);
 | |
|   svc_exit_timer=new QTimer(this);
 | |
|   connect(svc_exit_timer,SIGNAL(timeout()),this,SLOT(exitData()));
 | |
|   svc_exit_timer->start(100);
 | |
| 
 | |
|   if(!Startup(&err_msg)) {
 | |
|     Shutdown();
 | |
|     rda->syslog(LOG_ERR,"unable to start service component [%s]",
 | |
| 		(const char *)err_msg.toUtf8());
 | |
|     exit(3);
 | |
|   }
 | |
| 
 | |
|   //
 | |
|   // Maintenance Routine Timer
 | |
|   //
 | |
|   srandom(QTime::currentTime().msec());
 | |
|   svc_maint_timer=new QTimer(this);
 | |
|   svc_maint_timer->setSingleShot(true);
 | |
|   connect(svc_maint_timer,SIGNAL(timeout()),this,SLOT(checkMaintData()));
 | |
|   int interval=GetMaintInterval();
 | |
|   if(!rda->config()->disableMaintChecks()) {
 | |
|     svc_maint_timer->start(interval);
 | |
|   }
 | |
|   else {
 | |
|     rda->syslog(LOG_INFO,"maintenance checks disabled on this host");
 | |
|   }
 | |
| 
 | |
|   if(!RDWritePid(RD_PID_DIR,"rdservice.pid",getuid())) {
 | |
|     rda->syslog(LOG_WARNING,"unable to write pid file to \"%s/rdservice.pid\"",
 | |
| 		RD_PID_DIR);
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| void MainObject::processFinishedData(int id)
 | |
| {
 | |
|   svc_processes[id]->deleteLater();
 | |
|   svc_processes.remove(id);
 | |
| }
 | |
| 
 | |
| 
 | |
| void MainObject::exitData()
 | |
| {
 | |
|   QString err_msg;
 | |
| 
 | |
|   if(global_exiting) {
 | |
|     Shutdown();
 | |
|     RDDeletePid(RD_PID_DIR,"rdservice.pid");
 | |
|     rda->syslog(LOG_DEBUG,"shutting down normally");
 | |
|     exit(0);
 | |
|   }
 | |
| 
 | |
|   if(global_reload_dropboxes) {
 | |
|     ShutdownDropboxes();
 | |
|     StartDropboxes(&err_msg);
 | |
|     ::signal(SIGUSR1,SigHandler);
 | |
|     global_reload_dropboxes=false;
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| int main(int argc,char *argv[])
 | |
| {
 | |
|   QCoreApplication a(argc,argv);
 | |
|   new MainObject();
 | |
|   return a.exec();
 | |
| }
 |