mirror of
				https://github.com/ElvishArtisan/rivendell.git
				synced 2025-11-04 08:04:12 +01:00 
			
		
		
		
	* Cleaned up deprecation warnings for 'QString::sprintf()'. Signed-off-by: Fred Gleason <fredg@paravelsystems.com>
		
			
				
	
	
		
			400 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			400 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
// startup.cpp
 | 
						|
//
 | 
						|
// Startup routines for the Rivendell Services Manager
 | 
						|
//
 | 
						|
//   (C) Copyright 2018-2021 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 <stdio.h>
 | 
						|
#include <syslog.h>
 | 
						|
 | 
						|
#include <qstringlist.h>
 | 
						|
#include <qfileinfo.h>
 | 
						|
 | 
						|
#include <rdapplication.h>
 | 
						|
#include <rdconf.h>
 | 
						|
#include <rdescape_string.h>
 | 
						|
#include <rdpaths.h>
 | 
						|
 | 
						|
#include "rdservice.h"
 | 
						|
 | 
						|
bool MainObject::Startup(QString *err_msg)
 | 
						|
{
 | 
						|
  QStringList args;
 | 
						|
  QString sql;
 | 
						|
  RDSqlQuery *q;
 | 
						|
 | 
						|
  //
 | 
						|
  // Kill Stale Programs
 | 
						|
  //
 | 
						|
  KillProgram("rdrssd");
 | 
						|
  KillProgram("rdrepld");
 | 
						|
  KillProgram("rdvairplayd");
 | 
						|
  KillProgram("rdpadengined");
 | 
						|
  KillProgram("rdpadd");
 | 
						|
  KillProgram("rdcatchd");
 | 
						|
  KillProgram("ripcd");
 | 
						|
  KillProgram("caed");
 | 
						|
 | 
						|
  //
 | 
						|
  // caed(8)
 | 
						|
  //
 | 
						|
  svc_processes[RDSERVICE_CAED_ID]=new RDProcess(RDSERVICE_CAED_ID,this);
 | 
						|
  args.clear();
 | 
						|
  svc_processes[RDSERVICE_CAED_ID]->start(QString(RD_PREFIX)+"/sbin/caed",args);
 | 
						|
  if(!svc_processes[RDSERVICE_CAED_ID]->process()->waitForStarted(-1)) {
 | 
						|
    *err_msg=tr("unable to start caed(8)")+": "+
 | 
						|
      svc_processes[RDSERVICE_CAED_ID]->errorText();
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  if(svc_startup_target==MainObject::TargetCaed) {
 | 
						|
    fprintf(stderr,"Startup target caed(8) reached\n");
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  //
 | 
						|
  // ripcd(8)
 | 
						|
  //
 | 
						|
  svc_processes[RDSERVICE_RIPCD_ID]=new RDProcess(RDSERVICE_RIPCD_ID,this);
 | 
						|
  args.clear();
 | 
						|
  svc_processes[RDSERVICE_RIPCD_ID]->
 | 
						|
    start(QString(RD_PREFIX)+"/sbin/ripcd",args);
 | 
						|
  if(!svc_processes[RDSERVICE_RIPCD_ID]->process()->waitForStarted(-1)) {
 | 
						|
    *err_msg=tr("unable to start ripcd(8)")+": "+
 | 
						|
      svc_processes[RDSERVICE_RIPCD_ID]->errorText();
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  if(svc_startup_target==MainObject::TargetRipcd) {
 | 
						|
    fprintf(stderr,"Startup target ripcd(8) reached\n");
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  //
 | 
						|
  // rdcatchd(8)
 | 
						|
  //
 | 
						|
  svc_processes[RDSERVICE_RDCATCHD_ID]=new RDProcess(RDSERVICE_RDCATCHD_ID,this);
 | 
						|
  args.clear();
 | 
						|
  svc_processes[RDSERVICE_RDCATCHD_ID]->
 | 
						|
    start(QString(RD_PREFIX)+"/sbin/rdcatchd",args);
 | 
						|
  if(!svc_processes[RDSERVICE_RDCATCHD_ID]->process()->waitForStarted(-1)) {
 | 
						|
    *err_msg=tr("unable to start rdcatchd(8)")+": "+
 | 
						|
      svc_processes[RDSERVICE_RDCATCHD_ID]->errorText();
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  if(svc_startup_target==MainObject::TargetRdcatchd) {
 | 
						|
    fprintf(stderr,"Startup target rdcatchd(8) reached\n");
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  //
 | 
						|
  // rdpadd(8)
 | 
						|
  //
 | 
						|
  svc_processes[RDSERVICE_RDPADD_ID]=new RDProcess(RDSERVICE_RDPADD_ID,this);
 | 
						|
  args.clear();
 | 
						|
  svc_processes[RDSERVICE_RDPADD_ID]->
 | 
						|
    start(QString(RD_PREFIX)+"/sbin/rdpadd",args);
 | 
						|
  if(!svc_processes[RDSERVICE_RDPADD_ID]->process()->waitForStarted(-1)) {
 | 
						|
    *err_msg=tr("unable to start rdpadd(8)")+": "+
 | 
						|
      svc_processes[RDSERVICE_RDPADD_ID]->errorText();
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  if(svc_startup_target==MainObject::TargetRdpadd) {
 | 
						|
    fprintf(stderr,"Startup target rdpadd(8) reached\n");
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  //
 | 
						|
  // *** BAND-AID * BAND_AID * YEECH! ***
 | 
						|
  // This Makes It Work, but I think we're going to need to implement
 | 
						|
  // socket activation on all of these services.
 | 
						|
  //
 | 
						|
  sleep(1);
 | 
						|
 | 
						|
  //
 | 
						|
  // rdpadengined(8)
 | 
						|
  //
 | 
						|
  svc_processes[RDSERVICE_RDPADENGINED_ID]=
 | 
						|
    new RDProcess(RDSERVICE_RDPADENGINED_ID,this);
 | 
						|
  args.clear();
 | 
						|
  svc_processes[RDSERVICE_RDPADENGINED_ID]->
 | 
						|
    start(QString(RD_PREFIX)+"/sbin/rdpadengined",args);
 | 
						|
  if(!svc_processes[RDSERVICE_RDPADENGINED_ID]->process()->waitForStarted(-1)) {
 | 
						|
    *err_msg=tr("unable to start rdpadengined(8)")+": "+
 | 
						|
      svc_processes[RDSERVICE_RDPADENGINED_ID]->errorText();
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  if(svc_startup_target==MainObject::TargetRdpadengined) {
 | 
						|
    fprintf(stderr,"Startup target rdpadengined(8) reached\n");
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  //
 | 
						|
  // rdvairplayd(8)
 | 
						|
  //
 | 
						|
  svc_processes[RDSERVICE_RDVAIRPLAYD_ID]=new RDProcess(RDSERVICE_RDVAIRPLAYD_ID,this);
 | 
						|
  args.clear();
 | 
						|
  svc_processes[RDSERVICE_RDVAIRPLAYD_ID]->
 | 
						|
    start(QString(RD_PREFIX)+"/sbin/rdvairplayd",args);
 | 
						|
  if(!svc_processes[RDSERVICE_RDVAIRPLAYD_ID]->process()->waitForStarted(-1)) {
 | 
						|
    *err_msg=tr("unable to start rdvairplayd(8)")+": "+
 | 
						|
      svc_processes[RDSERVICE_RDVAIRPLAYD_ID]->errorText();
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  if(svc_startup_target==MainObject::TargetRdvairplayd) {
 | 
						|
    fprintf(stderr,"Startup target rdvairplayd(8) reached\n");
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  //
 | 
						|
  // rdrepld(8)
 | 
						|
  //
 | 
						|
  sql=QString("select `NAME` from `REPLICATORS` where ")+
 | 
						|
    "`STATION_NAME`='"+RDEscapeString(rda->station()->name())+"'";
 | 
						|
  q=new RDSqlQuery(sql);
 | 
						|
  if(q->first()) {
 | 
						|
    svc_processes[RDSERVICE_RDREPLD_ID]=new RDProcess(RDSERVICE_RDREPLD_ID,this);
 | 
						|
    args.clear();
 | 
						|
    svc_processes[RDSERVICE_RDREPLD_ID]->
 | 
						|
      start(QString(RD_PREFIX)+"/sbin/rdrepld",args);
 | 
						|
    if(!svc_processes[RDSERVICE_RDREPLD_ID]->process()->waitForStarted(-1)) {
 | 
						|
      *err_msg=tr("unable to start rdrepld(8)")+": "+
 | 
						|
	svc_processes[RDSERVICE_RDREPLD_ID]->errorText();
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  delete q;
 | 
						|
  if(svc_startup_target==MainObject::TargetRdrepld) {
 | 
						|
    fprintf(stderr,"Startup target rdrepld(8) reached\n");
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  //
 | 
						|
  // rdrssd(8)
 | 
						|
  //
 | 
						|
  sql=QString("select `RSS_PROCESSOR_STATION` from `SYSTEM`");
 | 
						|
  q=new RDSqlQuery(sql);
 | 
						|
  if(q->first()) {
 | 
						|
    if(q->value(0).toString().toLower()==rda->station()->name().toLower()) {
 | 
						|
      svc_processes[RDSERVICE_RDRSSD_ID]=
 | 
						|
	new RDProcess(RDSERVICE_RDRSSD_ID,this);
 | 
						|
      args.clear();
 | 
						|
      svc_processes[RDSERVICE_RDRSSD_ID]->
 | 
						|
	start(QString(RD_PREFIX)+"/sbin/rdrssd",args);
 | 
						|
      if(!svc_processes[RDSERVICE_RDRSSD_ID]->process()->waitForStarted(-1)) {
 | 
						|
	*err_msg=tr("unable to start rdrssd(8)")+": "+
 | 
						|
	  svc_processes[RDSERVICE_RDRSSD_ID]->errorText();
 | 
						|
	return false;
 | 
						|
      }
 | 
						|
    }
 | 
						|
    if(svc_startup_target==MainObject::TargetRdrssd) {
 | 
						|
      fprintf(stderr,"Startup target rdrssd(8) reached\n");
 | 
						|
      return true;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  delete q;
 | 
						|
 | 
						|
  if(!StartDropboxes(err_msg)) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  return true;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
bool MainObject::StartDropboxes(QString *err_msg)
 | 
						|
{
 | 
						|
  QString sql;
 | 
						|
  RDSqlQuery *q;
 | 
						|
  RDSqlQuery *q1;
 | 
						|
  int id=RDSERVICE_FIRST_DROPBOX_ID;
 | 
						|
 | 
						|
  //
 | 
						|
  // Launch Dropbox Configurations
 | 
						|
  //
 | 
						|
  sql=QString("select ")+
 | 
						|
    "`ID`,"+                       // 00
 | 
						|
    "`GROUP_NAME`,"+               // 01
 | 
						|
    "`PATH`,"+                     // 02
 | 
						|
    "`NORMALIZATION_LEVEL`,"+      // 03
 | 
						|
    "`AUTOTRIM_LEVEL`,"+           // 04
 | 
						|
    "`TO_CART`,"+                  // 05
 | 
						|
    "`USE_CARTCHUNK_ID`,"+         // 06
 | 
						|
    "`TITLE_FROM_CARTCHUNK_ID`,"+  // 07
 | 
						|
    "`DELETE_CUTS`,"+              // 08
 | 
						|
    "`METADATA_PATTERN`,"+         // 09
 | 
						|
    "`FIX_BROKEN_FORMATS`,"+       // 10
 | 
						|
    "`LOG_TO_SYSLOG`,"+            // 11
 | 
						|
    "`LOG_PATH`,"+                 // 12
 | 
						|
    "`DELETE_SOURCE`,"+            // 13
 | 
						|
    "`STARTDATE_OFFSET`,"+         // 14
 | 
						|
    "`ENDDATE_OFFSET`,"+           // 15
 | 
						|
    "`ID`,"+                       // 16
 | 
						|
    "`IMPORT_CREATE_DATES`,"+      // 17
 | 
						|
    "`CREATE_STARTDATE_OFFSET`,"+  // 18
 | 
						|
    "`CREATE_ENDDATE_OFFSET`,"+    // 19
 | 
						|
    "`SET_USER_DEFINED`,"+         // 20
 | 
						|
    "`FORCE_TO_MONO`,"+            // 21
 | 
						|
    "`SEGUE_LEVEL`,"+              // 22
 | 
						|
    "`SEGUE_LENGTH`,"+             // 23
 | 
						|
    "`SEND_EMAIL` "+               // 24
 | 
						|
    "from `DROPBOXES` where "+
 | 
						|
    "`STATION_NAME`='"+RDEscapeString(rda->config()->stationName())+"'";
 | 
						|
  q=new RDSqlQuery(sql);
 | 
						|
  while(q->next()) {
 | 
						|
    QStringList args;
 | 
						|
 | 
						|
    args.push_back(QString::asprintf("--persistent-dropbox-id=%d",
 | 
						|
				     q->value(16).toInt()));
 | 
						|
    args.push_back("--drop-box");
 | 
						|
    sql=QString("select `SCHED_CODE` from `DROPBOX_SCHED_CODES` where ")+
 | 
						|
      QString::asprintf("`DROPBOX_ID`=%d",q->value(0).toInt());
 | 
						|
    q1=new RDSqlQuery(sql);
 | 
						|
    while(q1->next()) {
 | 
						|
      args.push_back(QString::asprintf("--add-scheduler-code=")+
 | 
						|
		     q1->value(0).toString()+"");
 | 
						|
    }
 | 
						|
    delete q1;
 | 
						|
    args.push_back(QString::asprintf("--normalization-level=%d",
 | 
						|
				     q->value(3).toInt()/100));
 | 
						|
    args.push_back(QString::asprintf("--autotrim-level=%d",
 | 
						|
				     q->value(4).toInt()/100));
 | 
						|
    if(q->value(5).toUInt()>0) {
 | 
						|
      args.push_back(QString::asprintf("--to-cart=%u",q->value(5).toUInt()));
 | 
						|
    }
 | 
						|
    if(q->value(6).toString()=="Y") {
 | 
						|
      args.push_back("--use-cartchunk-cutid");
 | 
						|
    }
 | 
						|
    if(q->value(22).toInt()<1) {
 | 
						|
      args.push_back(QString::asprintf("--segue-level=%d",
 | 
						|
				       q->value(22).toInt()));
 | 
						|
      args.push_back(QString::asprintf("--segue-length=%u",
 | 
						|
				       q->value(23).toUInt()));
 | 
						|
    }
 | 
						|
    if(q->value(7).toString()=="Y") {
 | 
						|
      args.push_back("--title-from-cartchunk-cutid");
 | 
						|
    }
 | 
						|
    if(q->value(8).toString()=="Y") {
 | 
						|
      args.push_back("--delete-cuts");
 | 
						|
    }
 | 
						|
    if(q->value(21).toString()=="Y") {
 | 
						|
      args.push_back("--to-mono");
 | 
						|
    }
 | 
						|
    if(!q->value(9).toString().isEmpty()) {
 | 
						|
      args.push_back(QString("--metadata-pattern=")+q->value(9).toString());
 | 
						|
    }
 | 
						|
    if(q->value(10).toString()=="Y") {
 | 
						|
      args.push_back("--fix-broken-formats");
 | 
						|
    }
 | 
						|
    if(q->value(13).toString()=="Y") {
 | 
						|
      args.push_back("--delete-source");
 | 
						|
    }
 | 
						|
    if(q->value(24).toString()=="Y") {
 | 
						|
      args.push_back("--send-mail");
 | 
						|
      args.push_back("--mail-per-file");
 | 
						|
    }
 | 
						|
    if(q->value(17).toString()=="Y") {
 | 
						|
      args.push_back(QString::asprintf("--create-startdate-offset=%d",
 | 
						|
				       q->value(18).toInt()));
 | 
						|
      args.push_back(QString::asprintf("--create-enddate-offset=%d",
 | 
						|
				       q->value(19).toInt()));
 | 
						|
    }
 | 
						|
    if(!q->value(20).toString().isEmpty()) {
 | 
						|
      args.push_back(QString("--set-user-defined=")+q->value(20).toString());
 | 
						|
    }
 | 
						|
    args.push_back(QString::asprintf("--startdate-offset=%d",
 | 
						|
				     q->value(14).toInt()));
 | 
						|
    args.push_back(QString::asprintf("--enddate-offset=%d",
 | 
						|
				     q->value(15).toInt()));
 | 
						|
    if(RDBool(q->value(11).toString())) {
 | 
						|
      args.push_back("--log-syslog");
 | 
						|
    }
 | 
						|
    else {
 | 
						|
      if(!q->value(12).toString().isEmpty()) {
 | 
						|
	args.push_back("--log-filename="+q->value(12).toString());
 | 
						|
      }
 | 
						|
    }
 | 
						|
    args.push_back(q->value(1).toString());
 | 
						|
    args.push_back(q->value(2).toString());
 | 
						|
 | 
						|
    rda->syslog(LOG_DEBUG,"starting dropbox %d using command: \"%s\"",
 | 
						|
		q->value(0).toInt(),
 | 
						|
		(QString(RD_PREFIX)+"/bin/rdimport "+
 | 
						|
		 args.join(" ")).toUtf8().constData());
 | 
						|
    
 | 
						|
    svc_processes[id]=new RDProcess(id,this);
 | 
						|
    svc_processes[id]->start(QString(RD_PREFIX)+"/bin/rdimport",args);
 | 
						|
    if(!svc_processes[id]->process()->waitForStarted(-1)) {
 | 
						|
      *err_msg=tr("unable to start dropbox")+": "+
 | 
						|
	svc_processes[id]->errorText();
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
    id++;
 | 
						|
  }
 | 
						|
  delete q;
 | 
						|
  return true;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
void MainObject::KillProgram(const QString &program)
 | 
						|
{
 | 
						|
  QList<pid_t> pids=RDGetPids(program);
 | 
						|
 | 
						|
  while(pids.size()>0) {
 | 
						|
    for(int i=0;i<pids.size();i++) {
 | 
						|
      kill(pids.at(i),SIGKILL);
 | 
						|
      rda->syslog(LOG_WARNING,"killing unresponsive program \"%s\" [PID: %d]",
 | 
						|
		  (const char *)program.toUtf8(),pids.at(i));
 | 
						|
    }
 | 
						|
    sleep(1);
 | 
						|
    pids=RDGetPids(program);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
QString MainObject::TargetCommandString(MainObject::StartupTarget target) const
 | 
						|
{
 | 
						|
  switch(target) {
 | 
						|
  case MainObject::TargetCaed:
 | 
						|
    return QString("--end-startup-after-caed");
 | 
						|
 | 
						|
  case MainObject::TargetRipcd:
 | 
						|
    return QString("--end-startup-after-ripcd");
 | 
						|
 | 
						|
  case MainObject::TargetRdcatchd:
 | 
						|
    return QString("--end-startup-after-rdcatchd");
 | 
						|
 | 
						|
  case MainObject::TargetRdpadd:
 | 
						|
    return QString("--end-startup-after-rdpadd");
 | 
						|
 | 
						|
  case MainObject::TargetRdpadengined:
 | 
						|
    return QString("--end-startup-after-rdpadengined");
 | 
						|
 | 
						|
  case MainObject::TargetRdvairplayd:
 | 
						|
    return QString("--end-startup-after-rdvairplayd");
 | 
						|
 | 
						|
  case MainObject::TargetRdrepld:
 | 
						|
    return QString("--end-startup-after-rdrepld");
 | 
						|
 | 
						|
  case MainObject::TargetRdrssd:
 | 
						|
    return QString("--end-startup-after-rdrssd");
 | 
						|
 | 
						|
  case MainObject::TargetAll:
 | 
						|
    break;
 | 
						|
  }
 | 
						|
 | 
						|
  return QString();
 | 
						|
}
 |