好吧所以我正在研究一个用c ++编写的程序作为守护进程运行。它主要针对Linux用户,但我希望包括Windows(作为服务运行)和Mac用户。
我希望守护进程在手动关闭时进行记录。但是,由于系统暂停或重新启动,系统应该不记录关闭。
目前我已经屏蔽了所有信号并使用sigaction()实现了某种处理。在记录关闭之前,一个函数还检查系统的运行级别,如果它是0,1 0r 6,则省略日志记录。检查运行级别的方法是运行命令“runlevel”并处理输出。
我的问题是运行级别并不总是我所期望的。我正在运行Ubuntu,并且像往常一样登录时我处于运行级别2,重启时也是如此。停止时我有时从“运行级别”输出什么。不同的Linux发行版使用自己的运行级别,因此它不适合移植性。
那么有没有更好的方法来确定系统是否停止?此外,是否有更好的方法来捕获中断,例如通过异常处理等?
如果有任何帮助,我会粘贴一段代码。使用Poco C++ libraries以c ++编写。
void MainApplication::signalHandler(int sig) {
#if defined(POCO_OS_FAMILY_UNIX)
switch(sig) {
case -1:
struct sigaction act;
act.sa_handler = signalHandler;
sigemptyset(&act.sa_mask);
act.sa_flags = 0;
sigaction(SIGINT, &act, 0); //Better way to do this?
sigaction(SIGQUIT, &act, 0);
sigaction(SIGKILL, &act, 0);
sigaction(SIGTERM, &act, 0);
sigaction(SIGHUP, &act, 0);
sigaction(SIGSTOP, &act, 0);
sigaction(SIGTSTP, &act, 0);
sigaction(SIGCONT, &act, 0);
sigaction(SIGUSR1, &act, 0);
sigaction(SIGUSR2, &act, 0);
break;
case SIGINT:
case SIGQUIT:
case SIGTSTP:
case SIGHUP:
case SIGKILL:
case SIGTERM:
//Log Shutdown!
if (!isHalting())
_instance->_database->logBypass(BYPASS_SHUTDOWN);
terminateNicely(true);
break;
case SIGCONT:
//Continued, means stopped
break;
case SIGUSR1:
//Resetting Net Responsibility
_instance->uninitialize();
_instance->initialize(*_instance);
break;
case SIGUSR2:
//Other action or just mask it
break;
default:
//Caught signal
break;
}
#endif
}
bool MainApplication::isHalting() {
#if defined(POCO_OS_FAMILY_UNIX)
string cmd("runlevel");
vector<string> args;
Poco::Pipe outPipe;
ProcessHandle ph = Process::launch(cmd, args, 0, &outPipe, 0);
ph.wait();
Poco::PipeInputStream istr(outPipe);
stringstream ss;
Poco::StreamCopier::copyStream(istr, ss);
int runlevel;
ss.seekg(-2, ios::end);
ss >> runlevel;
return (runlevel == 0 || runlevel == 1 || runlevel == 6);
#else
return false;
#endif
}
答案 0 :(得分:0)
我最终走另一条路来解决问题。现在我正在阅读计算机的启动历史记录,并将其与软件历史记录进行比较,以查看计算机是否在没有软件的情况下启动或暂停。这就足够了,可能更准确。
用于确定Linux中启动历史记录的命令是
$ last reboot