2022-09-22 20:57:46 -04:00
|
|
|
// This file is part of Motion Watch.
|
2022-04-14 09:45:54 -04:00
|
|
|
|
2022-09-22 20:57:46 -04:00
|
|
|
// Motion Watch 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.
|
2022-04-14 09:45:54 -04:00
|
|
|
|
2022-09-22 20:57:46 -04:00
|
|
|
// Motion Watch 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.
|
2022-04-14 09:45:54 -04:00
|
|
|
|
2022-09-22 20:57:46 -04:00
|
|
|
#include "mo_detect.h"
|
2022-12-11 10:25:22 -05:00
|
|
|
#include "logger.h"
|
2022-08-12 21:46:36 -04:00
|
|
|
|
2023-02-05 14:05:56 -05:00
|
|
|
void detectMoInFile(const string &detPath, const string &recPath, shared_t *share)
|
2022-04-14 09:45:54 -04:00
|
|
|
{
|
2022-12-21 20:30:37 -05:00
|
|
|
detLog("detect_mo_in_file() -- start", share);
|
2022-12-11 10:25:22 -05:00
|
|
|
|
2022-12-21 20:30:37 -05:00
|
|
|
Mat thumbNail;
|
2022-04-14 09:45:54 -04:00
|
|
|
|
2023-02-05 14:05:56 -05:00
|
|
|
if (moDetect(detPath, thumbNail, share))
|
2022-08-12 21:46:36 -04:00
|
|
|
{
|
2022-12-21 20:30:37 -05:00
|
|
|
share->skipCmd = true;
|
2022-08-12 21:46:36 -04:00
|
|
|
|
2023-02-05 14:05:56 -05:00
|
|
|
wrOut(recPath, thumbNail, share);
|
2022-12-21 20:30:37 -05:00
|
|
|
}
|
2023-02-05 14:05:56 -05:00
|
|
|
|
|
|
|
if (exists(detPath)) remove(detPath);
|
|
|
|
if (exists(recPath)) remove(recPath);
|
|
|
|
|
|
|
|
detLog("detect_mo_in_file() -- finished", share);
|
|
|
|
}
|
|
|
|
|
2023-02-07 23:19:41 -05:00
|
|
|
static void *runCmd(void *arg)
|
2023-02-05 14:05:56 -05:00
|
|
|
{
|
2023-02-07 23:19:41 -05:00
|
|
|
auto args = static_cast<cmdRes_t*>(arg);
|
2023-02-05 14:05:56 -05:00
|
|
|
|
2023-02-07 23:19:41 -05:00
|
|
|
args->finished = false;
|
|
|
|
args->ret = 0;
|
2023-02-05 14:05:56 -05:00
|
|
|
|
2023-02-07 23:19:41 -05:00
|
|
|
pthread_setcancelstate(PTHREAD_CANCEL_ENABLE, NULL);
|
|
|
|
|
|
|
|
args->ret = pclose(popen(args->cmd.c_str(), "r"));
|
|
|
|
args->finished = true;
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool allCmdsFinished(const vector<cmdRes_t*> &resList)
|
|
|
|
{
|
|
|
|
for (auto &&res : resList)
|
|
|
|
{
|
|
|
|
if (!res->finished) return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2023-02-05 14:05:56 -05:00
|
|
|
}
|
|
|
|
|
2023-02-07 23:19:41 -05:00
|
|
|
void cancelAllCmds(const vector<cmdRes_t*> &resList)
|
2023-02-05 14:05:56 -05:00
|
|
|
{
|
2023-02-07 23:19:41 -05:00
|
|
|
for (auto &&res : resList)
|
|
|
|
{
|
|
|
|
if (!res->finished)
|
|
|
|
{
|
|
|
|
pthread_cancel(res->thr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-02-05 14:05:56 -05:00
|
|
|
|
2023-02-07 23:19:41 -05:00
|
|
|
bool allCmdsDidNotFail(const vector<cmdRes_t*> &resList)
|
|
|
|
{
|
|
|
|
for (auto &&res : resList)
|
2022-12-21 20:30:37 -05:00
|
|
|
{
|
2023-02-07 23:19:41 -05:00
|
|
|
if (res->ret != 0) return false;
|
2022-04-14 09:45:54 -04:00
|
|
|
}
|
2022-12-11 10:25:22 -05:00
|
|
|
|
2023-02-07 23:19:41 -05:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void cleanupRes(vector<cmdRes_t*> &resList)
|
|
|
|
{
|
|
|
|
for (auto &&res : resList)
|
|
|
|
{
|
|
|
|
delete res;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void runTOCmds(int timeOut, const vector<string> &cmds, vector<cmdRes_t*> &resList, bool abs = true)
|
|
|
|
{
|
|
|
|
resList.clear();
|
|
|
|
|
|
|
|
for (auto &&cmd : cmds)
|
|
|
|
{
|
|
|
|
auto res = new struct cmdRes_t;
|
|
|
|
|
|
|
|
res->cmd = cmd;
|
|
|
|
res->finished = false;
|
|
|
|
res->ret = pthread_create(&res->thr, NULL, &runCmd, res);
|
|
|
|
|
|
|
|
resList.push_back(res);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (abs)
|
|
|
|
{
|
|
|
|
sleep(timeOut);
|
|
|
|
|
|
|
|
if (!allCmdsFinished(resList))
|
|
|
|
{
|
|
|
|
sleep(2);
|
|
|
|
}
|
|
|
|
|
|
|
|
cancelAllCmds(resList);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (auto i = 0; !allCmdsFinished(resList); ++i)
|
|
|
|
{
|
|
|
|
sleep(1);
|
|
|
|
|
|
|
|
if (i >= timeOut)
|
|
|
|
{
|
|
|
|
cancelAllCmds(resList); break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void runTOCmd(int timeout, const string &cmd, bool abs = true)
|
|
|
|
{
|
|
|
|
vector<string> cmds;
|
|
|
|
vector<cmdRes_t*> rets;
|
|
|
|
|
|
|
|
cmds.push_back(cmd);
|
|
|
|
|
|
|
|
runTOCmds(timeout, cmds, rets, abs);
|
|
|
|
cleanupRes(rets);
|
2022-07-08 15:24:45 -04:00
|
|
|
}
|
|
|
|
|
2022-09-22 20:57:46 -04:00
|
|
|
void recLoop(shared_t *share)
|
2022-08-12 21:46:36 -04:00
|
|
|
{
|
2022-09-22 20:57:46 -04:00
|
|
|
while (rdConf(share))
|
2022-08-12 21:46:36 -04:00
|
|
|
{
|
2022-12-13 20:27:32 -05:00
|
|
|
enforceMaxLogSize(share->recLogPath, share);
|
|
|
|
enforceMaxLogSize(share->detLogPath, share);
|
|
|
|
|
|
|
|
initLogFile(share->recLogPath, share->recLogFile);
|
|
|
|
initLogFile(share->detLogPath, share->detLogFile);
|
|
|
|
|
|
|
|
initLogFrontPages(share);
|
|
|
|
|
2023-02-11 20:59:19 -05:00
|
|
|
recLog("rec_loop() -- start", share);
|
|
|
|
|
|
|
|
if (!exists("/tmp/mow-lock") && share->updateRoot)
|
2022-12-04 15:13:39 -05:00
|
|
|
{
|
|
|
|
system("touch /tmp/mow-lock");
|
2022-12-13 20:27:32 -05:00
|
|
|
|
2022-12-11 10:25:22 -05:00
|
|
|
genCSS(share);
|
|
|
|
genHTMLul(share->webRoot, string(APP_NAME) + " " + string(APP_VER), share);
|
2022-12-13 20:27:32 -05:00
|
|
|
|
|
|
|
remove("/tmp/mow-lock");
|
2022-12-16 18:24:18 -05:00
|
|
|
recLog("webroot page updated: " + cleanDir(share->webRoot) + "/index.html", share);
|
2023-02-11 20:59:19 -05:00
|
|
|
|
|
|
|
share->updateRoot = false;
|
2022-12-11 10:25:22 -05:00
|
|
|
}
|
2022-09-23 21:50:06 -04:00
|
|
|
|
2022-12-11 12:33:56 -05:00
|
|
|
genHTMLul(share->outDir, share->camName, share);
|
2022-12-13 20:27:32 -05:00
|
|
|
|
2022-12-16 18:24:18 -05:00
|
|
|
recLog("camera specific webroot page updated: " + share->outDir + "/index.html", share);
|
2022-12-11 12:33:56 -05:00
|
|
|
|
2022-12-24 13:48:51 -05:00
|
|
|
for (auto i = 0; i < share->numOfClips; ++i)
|
2022-12-21 20:30:37 -05:00
|
|
|
{
|
2023-02-05 14:05:56 -05:00
|
|
|
auto detPath = cleanDir(share->buffDir) + "/" + to_string(i) + share->detSuffix;
|
|
|
|
auto recPath = cleanDir(share->buffDir) + "/" + to_string(i) + share->recSuffix;
|
2022-12-22 20:15:14 -05:00
|
|
|
|
2023-02-07 23:19:41 -05:00
|
|
|
vector<string> cmds;
|
|
|
|
vector<cmdRes_t*> rets;
|
|
|
|
|
2023-02-05 14:05:56 -05:00
|
|
|
if (share->recordUrl == share->detectUrl)
|
|
|
|
{
|
|
|
|
recPath = detPath;
|
|
|
|
}
|
2023-02-07 23:19:41 -05:00
|
|
|
else
|
|
|
|
{
|
|
|
|
cmds.push_back("ffmpeg -hide_banner -i " + share->recordUrl + " -y -vcodec " + share->vidCodec + " -t " + to_string(share->clipLen) + " " + recPath);
|
|
|
|
}
|
2022-08-12 21:46:36 -04:00
|
|
|
|
2023-02-07 23:19:41 -05:00
|
|
|
cmds.push_back("ffmpeg -hide_banner -i " + share->detectUrl + " -y -vcodec " + share->vidCodec + " -t " + to_string(share->clipLen) + " " + detPath);
|
2022-08-12 21:46:36 -04:00
|
|
|
|
2023-02-07 23:19:41 -05:00
|
|
|
recLog("fetching camera footage -- ", share);
|
2022-12-13 20:27:32 -05:00
|
|
|
|
2023-02-07 23:19:41 -05:00
|
|
|
runTOCmds(share->clipLen, cmds, rets);
|
2022-12-11 10:25:22 -05:00
|
|
|
|
2023-02-07 23:19:41 -05:00
|
|
|
if (allCmdsDidNotFail(rets))
|
2022-12-21 20:30:37 -05:00
|
|
|
{
|
2023-02-05 14:05:56 -05:00
|
|
|
new thread(detectMoInFile, detPath, recPath, share);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-02-07 23:19:41 -05:00
|
|
|
recLog("one or both fetch cmds failed.", share);
|
|
|
|
}
|
|
|
|
|
|
|
|
recLog("fetch results:", share);
|
|
|
|
|
|
|
|
for (auto &&ret : rets)
|
|
|
|
{
|
|
|
|
recLog("cmd: " + ret->cmd + " return_code: " + to_string(ret->ret), share);
|
2022-12-21 20:30:37 -05:00
|
|
|
}
|
|
|
|
}
|
2022-07-28 10:30:07 -04:00
|
|
|
|
2022-09-22 20:57:46 -04:00
|
|
|
if (!share->skipCmd)
|
2022-08-12 21:46:36 -04:00
|
|
|
{
|
2022-12-21 20:30:37 -05:00
|
|
|
recLog("no motion detected", share);
|
2022-12-13 20:27:32 -05:00
|
|
|
|
|
|
|
if (share->postCmd.empty())
|
|
|
|
{
|
|
|
|
recLog("post command not defined, skipping.", share);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-12-16 18:24:18 -05:00
|
|
|
recLog("running post command: " + share->postCmd, share);
|
2023-02-07 23:19:41 -05:00
|
|
|
runTOCmd(14, share->postCmd, false);
|
2022-12-13 20:27:32 -05:00
|
|
|
}
|
2022-08-12 21:46:36 -04:00
|
|
|
}
|
2022-12-11 10:25:22 -05:00
|
|
|
else
|
|
|
|
{
|
2022-12-21 20:30:37 -05:00
|
|
|
recLog("motion detected, skipping the post command.", share);
|
2022-12-11 10:25:22 -05:00
|
|
|
}
|
|
|
|
|
2022-12-16 18:24:18 -05:00
|
|
|
recLog("rec_loop() -- finished", share);
|
2022-12-11 10:25:22 -05:00
|
|
|
|
2022-12-13 20:27:32 -05:00
|
|
|
if (share->retCode != 0)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2022-07-08 15:24:45 -04:00
|
|
|
}
|
|
|
|
|
2023-02-11 20:59:19 -05:00
|
|
|
void sigHandler(int signal, siginfo_t *info, void *context)
|
|
|
|
{
|
|
|
|
cerr << "received signal details: " << endl;
|
|
|
|
cerr << "-- si_pid: " << info->si_pid << endl;
|
|
|
|
cerr << "-- si_signo: " << info->si_signo << endl;
|
|
|
|
cerr << "-- si_code: " << info->si_code << endl;
|
|
|
|
cerr << "-- si_errno: " << info->si_errno << endl;
|
|
|
|
cerr << "-- si_uid: " << info->si_uid << endl;
|
|
|
|
cerr << "-- si_addr: " << info->si_addr << endl;
|
|
|
|
cerr << "-- si_status: " << info->si_status << endl;
|
|
|
|
cerr << "-- si_band: " << info->si_band << endl;
|
|
|
|
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
2022-07-08 15:24:45 -04:00
|
|
|
int main(int argc, char** argv)
|
|
|
|
{
|
2022-09-22 20:57:46 -04:00
|
|
|
struct shared_t sharedRes;
|
|
|
|
|
2023-02-11 20:59:19 -05:00
|
|
|
struct sigaction act = { 0 };
|
|
|
|
|
|
|
|
act.sa_flags = SA_SIGINFO;
|
|
|
|
act.sa_sigaction = &sigHandler;
|
|
|
|
|
|
|
|
sigaction(SIGTERM, &act, NULL);
|
|
|
|
sigaction(SIGABRT, &act, NULL);
|
|
|
|
sigaction(SIGSEGV, &act, NULL);
|
|
|
|
sigaction(SIGILL, &act, NULL);
|
|
|
|
sigaction(SIGFPE, &act, NULL);
|
|
|
|
sigaction(SIGBUS, &act, NULL);
|
|
|
|
|
2022-12-24 13:48:51 -05:00
|
|
|
sharedRes.conf = parseForList("-c", argc, argv);
|
2022-07-08 15:24:45 -04:00
|
|
|
|
|
|
|
if (parseForParam("-h", argc, argv, true) == "true")
|
2022-04-14 09:45:54 -04:00
|
|
|
{
|
2022-09-23 21:50:06 -04:00
|
|
|
cout << "Motion Watch " << APP_VER << endl << endl;
|
2022-09-22 20:57:46 -04:00
|
|
|
cout << "Usage: mow <argument>" << endl << endl;
|
|
|
|
cout << "-h : display usage information about this application." << endl;
|
2022-12-24 13:48:51 -05:00
|
|
|
cout << "-c : path to a config file." << endl;
|
|
|
|
cout << "-v : display the current version." << endl << endl;
|
|
|
|
cout << "note: multiple -c config files can be passed, reading from left" << endl;
|
|
|
|
cout << " to right. any conflicting values between the files will" << endl;
|
|
|
|
cout << " have the latest value from the latest file overwrite the" << endl;
|
|
|
|
cout << " the earliest." << endl;
|
2022-09-23 21:50:06 -04:00
|
|
|
}
|
2022-12-04 15:13:39 -05:00
|
|
|
else if (parseForParam("-v", argc, argv, true) == "true")
|
2022-09-23 21:50:06 -04:00
|
|
|
{
|
|
|
|
cout << APP_VER << endl;
|
2022-04-14 09:45:54 -04:00
|
|
|
}
|
2022-07-08 15:24:45 -04:00
|
|
|
else if (sharedRes.conf.empty())
|
2022-04-14 09:45:54 -04:00
|
|
|
{
|
2022-12-24 13:48:51 -05:00
|
|
|
cerr << "err: no config file(s) were given in -c" << endl;
|
2022-04-14 09:45:54 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-02-11 20:59:19 -05:00
|
|
|
sharedRes.retCode = 0;
|
|
|
|
sharedRes.updateRoot = true;
|
|
|
|
sharedRes.skipCmd = false;
|
|
|
|
sharedRes.init = true;
|
2022-12-11 10:25:22 -05:00
|
|
|
|
2022-12-04 15:13:39 -05:00
|
|
|
recLoop(&sharedRes);
|
2022-04-14 09:45:54 -04:00
|
|
|
|
2022-07-08 15:24:45 -04:00
|
|
|
return sharedRes.retCode;
|
2022-04-14 09:45:54 -04:00
|
|
|
}
|
|
|
|
|
2022-07-08 15:24:45 -04:00
|
|
|
return EINVAL;
|
2022-04-14 09:45:54 -04:00
|
|
|
}
|