-
Notifications
You must be signed in to change notification settings - Fork 0
/
mycMotDetRecPyC.cpp
1069 lines (923 loc) · 50.2 KB
/
mycMotDetRecPyC.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
/* PUT INIReader.h in same directory as this C++ program mycMotDetRecPy.cpp
* You find INIReader.h at https://github.com/benhoyt/inih/blob/master/cpp/INIReader.h
*
* PUT the MASK file "test_mask.png" (or your own mask file with any other name like "mask_stream2.png" also in the same directory
*
* INSTALL the necessary compilation libraries for g++ and others
*
* sudo apt update
* sudo apt upgrade
* sudo apt-get install build-essential cmake pkg-config libjpeg-dev libtiff5-dev libjasper-dev libpng-dev libavcodec-dev libavformat-dev libswscale-dev libv4l-dev libxvidcore-dev libx264-dev libfontconfig1-dev libcairo2-dev libgdk-pixbuf2.0-dev libpango1.0-dev libgtk2.0-dev libgtk-3-dev libatlas-base-dev gfortran libhdf5-dev libhdf5-serial-dev libhdf5-103 python3-pyqt5 python3-dev -y
* sudo apt-get install libcurl4-gnutls-dev
* sudo apt-get install libjsoncpp-dev
* sudo ln -s /usr/include/jsoncpp/json/ /usr/include/json
* sudo apt-get install libcurl4-openssl-dev openssl libcrypto
* Probably needed as well to suppress messages Install only when they appear: sudo apt-get install libatk-adaptor libgail-common
*
* COMPILE NOW THIS PROGRAM WITH THIS COMMAND, ASSUMING YOU HAVE INSTALLED ALL LIBs op opencv, libcur and jsonccp
* COMPILE WITH THIS WITH g++ mycMotDetRecPyC.cpp -o mycMotDetRecPyC $(python3-config --ldflags --cflags --embed) -I/usr/include/opencv4 -I/usr/include -lopencv_videoio -lopencv_video -lopencv_videostab -lopencv_core -lopencv_imgproc -lopencv_objdetect -lopencv_highgui -lopencv_imgcodecs -ljsoncpp -lcurl -lssl -lcrypto
*
* */
#define PY_SSIZE_T_CLEAN
#define _GLIBCXX_USE_NANOSLEEP
#include <future>
#include <thread>
#include <Python.h>
#include <iostream>
#include <string>
#include <opencv4/opencv2/opencv.hpp>
#include <opencv4/opencv2/imgproc.hpp>
#include <opencv4/opencv2/videoio.hpp>
#include <opencv4/opencv2/highgui.hpp>
#include <opencv4/opencv2/video.hpp>
#include "./INIReader.h"
#include <curl/curl.h>
#include <json/json.h>
#include <unistd.h>
#include <sys/mman.h>
#include <sys/wait.h>
#include <fcntl.h>
#include <cstring>
#include <semaphore.h>
#include <signal.h>
#include <queue>
#include <chrono>
#include <iostream>
#include <fstream>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <openssl/ssl.h>
#include <openssl/err.h>
#include <openssl/bio.h>
#include <sstream>
#include <netdb.h>
using namespace cv;
using namespace std;
const int MAIL_BUFFER_SIZE = 5120;
string base64_encode(const string &input) {
BIO *bio, *b64;
BUF_MEM *bufferPtr;
b64 = BIO_new(BIO_f_base64());
BIO_set_flags(b64, BIO_FLAGS_BASE64_NO_NL);
bio = BIO_new(BIO_s_mem());
bio = BIO_push(b64, bio);
BIO_write(bio, input.c_str(), input.length());
BIO_flush(bio);
BIO_get_mem_ptr(bio, &bufferPtr);
string result(bufferPtr->data, bufferPtr->length);
BIO_free_all(bio);
return result;
}
string buildAttachment(const string &filename, const string &data) {
stringstream attachment;
attachment << "--boundary\r\n";
attachment << "Content-Type: image/jpeg\r\n";
attachment << "Content-Disposition: attachment; filename=\"" << filename << "\"\r\n";
attachment << "Content-Transfer-Encoding: base64\r\n\r\n";
attachment << base64_encode(data) << "\r\n";
return attachment.str();
}
void sendEmails(string show_email_status_msg_on_display_console, string mail_email_pictures,
const char* smtpServer, const int smtpPort, const char* username, const char* password,
const char* senderEmail, const char* recipientEmail, const char* subject, const char* body,
const string attachment_file_fullpathname, const string attachment_filename ) {
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** MAIL INFORMATION" << endl;
cout << "*** Storage pathname of the attachment file : " << attachment_file_fullpathname << endl;
cout << "*** The name of the embedded attachment file: " << attachment_filename << endl;
cout << "\n*** Get SMTP server IP address" << endl;
}
// Get SMTP server IP address
struct hostent *server = gethostbyname(smtpServer);
if (server == NULL) {
cerr << "Error: Failed to resolve SMTP server hostname: " << smtpServer << endl;
return;
}
char* smtpServerIP = inet_ntoa(*((struct in_addr*) server->h_addr));
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Initialize OpenSSL" << endl;
}
// Initialize OpenSSL
SSL_library_init();
SSL_CTX *ssl_ctx = SSL_CTX_new(TLS_client_method());
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Create socket" << endl;
}
// Create socket
int clientSocket = socket(AF_INET, SOCK_STREAM, 0);
if (clientSocket == -1) {
cerr << "Error: Failed to create socket." << endl;
return;
}
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Connect to SMTP server without SSL" << endl;
}
// Connect to SMTP server without SSL
struct sockaddr_in serverAddress;
serverAddress.sin_family = AF_INET;
serverAddress.sin_port = htons(smtpPort); // SMTP port
serverAddress.sin_addr.s_addr = inet_addr(smtpServerIP);
if (connect(clientSocket, (struct sockaddr *)&serverAddress, sizeof(serverAddress)) == -1) {
cerr << "Error: Connection to SMTP server failed." << endl;
close(clientSocket);
return;
}
// Receive initial response from server
char buffer[MAIL_BUFFER_SIZE];
recv(clientSocket, buffer, MAIL_BUFFER_SIZE, 0);
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Received initial response from server" << endl;
cout << buffer << endl;
}
// Send EHLO command
string ehloCommand = "EHLO client\r\n";
send(clientSocket, ehloCommand.c_str(), ehloCommand.size(), 0);
recv(clientSocket, buffer, MAIL_BUFFER_SIZE, 0);
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Response after send EHLO command" << endl;
cout << buffer << endl;
}
// Send STARTTLS command
string starttlsCommand = "STARTTLS\r\n";
send(clientSocket, starttlsCommand.c_str(), starttlsCommand.size(), 0);
recv(clientSocket, buffer, MAIL_BUFFER_SIZE, 0);
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Response after send STARTTLS command" << endl;
cout << buffer << endl;
}
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Start SSL/TLS" << endl;
}
// Start SSL/TLS
SSL *ssl = SSL_new(ssl_ctx);
SSL_set_fd(ssl, clientSocket);
if (SSL_connect(ssl) == -1) {
cerr << "Error: SSL connection failed." << endl;
SSL_free(ssl);
close(clientSocket);
return;
}
// Send EHLO command over SSL
string ehloCommandSSL = "EHLO client\r\n";
SSL_write(ssl, ehloCommandSSL.c_str(), ehloCommandSSL.size());
SSL_read(ssl, buffer, MAIL_BUFFER_SIZE);
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Response after send EHLO command over SSL" << endl;
cout << buffer << endl;
}
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Send AUTH LOGIN command" << endl;
}
// Send AUTH LOGIN command
string authLoginCommand = "AUTH LOGIN\r\n";
SSL_write(ssl, authLoginCommand.c_str(), authLoginCommand.size());
SSL_read(ssl, buffer, MAIL_BUFFER_SIZE);
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Response after send AUTH LOGIN command" << endl;
cout << buffer << endl;
}
// Send base64 encoded username
string usernameBase64 = base64_encode(username);
SSL_write(ssl, (usernameBase64 + "\r\n").c_str(), usernameBase64.size() + 2);
SSL_read(ssl, buffer, MAIL_BUFFER_SIZE);
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Response after send base64 encoded username" << endl;
cout << buffer << endl;
}
// Send base64 encoded password
string passwordBase64 = base64_encode(password);
SSL_write(ssl, (passwordBase64 + "\r\n").c_str(), passwordBase64.size() + 2);
SSL_read(ssl, buffer, MAIL_BUFFER_SIZE);
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Response after send base64 encoded password" << endl;
cout << buffer << endl;
}
// Send MAIL FROM command
string mailFromCommand = "MAIL FROM: <" + string(senderEmail) + ">\r\n";
SSL_write(ssl, mailFromCommand.c_str(), mailFromCommand.size());
SSL_read(ssl, buffer, MAIL_BUFFER_SIZE);
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Response after send MAIL FROM command" << endl;
cout << buffer << endl;
}
// Send RCPT TO command
string rcptToCommand = "RCPT TO: <" + string(recipientEmail) + ">\r\n";
SSL_write(ssl, rcptToCommand.c_str(), rcptToCommand.size());
SSL_read(ssl, buffer, MAIL_BUFFER_SIZE);
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Response after send RCPT TO command" << endl;
cout << buffer << endl;
}
// Send DATA command
string dataCommand = "DATA\r\n";
SSL_write(ssl, dataCommand.c_str(), dataCommand.size());
SSL_read(ssl, buffer, MAIL_BUFFER_SIZE);
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Response after send DATA command" << endl;
cout << buffer << endl;
}
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Send email headers and body" << endl;
}
// Send email headers and body
string emailData = "From: " + string(senderEmail) + "\r\n";
emailData += "To: " + string(recipientEmail) + "\r\n";
emailData += "Subject: " + string(subject) + "\r\n";
if (mail_email_pictures == "Yes") {
emailData += "Content-Type: multipart/mixed; boundary=boundary\r\n\r\n";
emailData += "--boundary\r\n";
emailData += "Content-Type: text/plain\r\n\r\n";
emailData += string(body) + " The attached file was saved at: " + attachment_file_fullpathname + "\r\n";
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Read attachment file and append attachment to email data" << endl;
}
// Read attachment file and append attachment to email data
ifstream file(attachment_file_fullpathname, ios::binary);
if (file) {
string attachment((istreambuf_iterator<char>(file)), (istreambuf_iterator<char>()));
emailData += buildAttachment(attachment_filename, attachment);
file.close();
} else {
cerr << "Error: Failed to open JPG file." << endl;
SSL_free(ssl);
close(clientSocket);
return;
}
// End MIME boundary
emailData += "--boundary--\r\n";
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** End MIME boundary" << endl;
}
}
// End of DATA
emailData += "\r\n.\r\n";
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** End of DATA" << endl;
}
// Send email data
SSL_write(ssl, emailData.c_str(), emailData.size());
SSL_read(ssl, buffer, MAIL_BUFFER_SIZE);
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Response after send email data" << endl;
cout << buffer << endl;
}
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Send QUIT command" << endl;
}
// Send QUIT command
string quitCommand = "QUIT\r\n";
SSL_write(ssl, quitCommand.c_str(), quitCommand.size());
SSL_read(ssl, buffer, MAIL_BUFFER_SIZE);
// cout << buffer << endl;
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Close SSL connection and free SSL context" << endl;
}
// Close SSL connection and free SSL context
SSL_shutdown(ssl);
SSL_free(ssl);
SSL_CTX_free(ssl_ctx);
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** Close socket" << endl;
}
// Close socket
close(clientSocket);
if (show_email_status_msg_on_display_console == "Yes") {
cout << "*** End of mail function" << endl;
}
return;
}
VideoWriter outputVideo;
// Define the function to be called when ctrl-c (SIGINT) is sent to process
void signal_callback_handler(int signum) {
cout << "Caught signal CTRL-c. Program stops." << endl;
// Terminate program
exit(signum);
}
// Function to split a string by delimiter
vector<string> splitString(const string& input, char delimiter) {
vector<string> tokens;
string token;
istringstream tokenStream(input);
while (getline(tokenStream, token, delimiter)) {
tokens.push_back(token);
}
return tokens;
}
// Structure to hold object detection result
struct DetectionResult {
string label;
string confidence;
cv::Rect boundingBox;
};
const int BUFFER_SIZE = 2048;
struct SharedMemory1 {
char message1[BUFFER_SIZE];
sem_t mutex1;
};
struct SharedMemory2 {
char message2[BUFFER_SIZE];
sem_t mutex2;
};
void getTapoMessages(SharedMemory1* sharedMemory1, const char* moduleName, const char* functionName) {
PyObject *pModuleName, *pModule, *pFunc;
PyObject *pArgs, *pTupleValue, *pValue;
const char* pFunctionName;
if (Py_IsInitialized() == false) {
Py_Initialize();
PyRun_SimpleString("import sys");
PyRun_SimpleString("sys.path.append(\".\")");
}
/* Error checking of pName left out */
pModuleName = PyUnicode_DecodeFSDefault(moduleName);
pFunctionName = functionName;
pModule = PyImport_Import(pModuleName);
Py_DECREF(pModuleName);
const char* savedString;
if (pModule != NULL) {
pFunc = PyObject_GetAttrString(pModule, pFunctionName);
/* pFunc is a new reference */
pArgs = PyTuple_New(1); // 1 means the length of the tuple!
if (pFunc && PyCallable_Check(pFunc)) {
// here we define a first tuple value, we pass the tuple to the function
pTupleValue = PyUnicode_DecodeFSDefault("my test message");
PyTuple_SetItem(pArgs, 0, pTupleValue);
while (true) {
pValue = PyObject_CallObject(pFunc, pArgs);
if (pValue != NULL) {
// Convert the result to a C string
const char* resultString = PyUnicode_AsUTF8(pValue);
if (resultString) {
// Save the string value in a C++ variable
savedString = resultString;
// Print the saved string value
// cout << "Saved string value: " << savedString << endl;
//cout << savedString << endl;
// Write result to shared memory
sem_wait(&sharedMemory1->mutex1);
snprintf(sharedMemory1->message1, BUFFER_SIZE, "%s", savedString);
sem_post(&sharedMemory1->mutex1);
//sleep(1); // slow down the speed 0.3 seconds, too fast leads to too much errors
}
Py_DECREF(pValue);
}
else {
Py_DECREF(pFunc);
Py_DECREF(pModule);
PyErr_Print();
fprintf(stderr,"Call failed\n");
//return;
}
} // END of while (true)
}
else {
if (PyErr_Occurred())
PyErr_Print();
fprintf(stderr, "Cannot find the Python function (default is: getTapoEventMessage). Check .ini file.\n");
cout << "Looking for: " << functionName << endl;
}
Py_XDECREF(pFunc);
Py_DECREF(pModule);
}
else {
PyErr_Print();
fprintf(stderr, "Failed to load the Python module (default is: myTapoEvents). Check .ini file.\n");
cout << "Looking for: " << moduleName << ".py" << endl;
}
}
size_t write_data(void *ptr, size_t size, size_t nmemb, string *data) {
data->append((char *)ptr, size * nmemb);
return size * nmemb;
}
void postImageAndGetResponse(SharedMemory2* sharedMemory2, string& AIserverUrl, string& min_confidence, Mat& frame, string& show_AIResponse_message, string& show_AIObjDetectionResult, string& curl_debug_message_on) {
CURL *curl;
CURLcode res;
string response_data;
curl = curl_easy_init();
if (curl) {
if (curl_debug_message_on == "Yes") {
curl_easy_setopt(curl, CURLOPT_VERBOSE, 1L);
}
// Convert the frame to JPEG format
vector<uchar> buffer;
//buffer for coding
//vector<int> param(2);
//param[0] = cv::IMWRITE_JPEG_QUALITY; param[1] = 100; //default(95) 0-100
// cv::imencode(".jpg", frame, buffer, param);
cv::imencode(".jpg", frame, buffer);
// Save the frame into a file
// imwrite("./temp_frame.jpg", frame); //
// Mat frame = imread("./temp_frame.jpg"); // Load your image here.
// string AIserverUrl = AIserverUrl; // set in the config file and passsed by main()
// string min_confidence = "0.4"; // set in the config file and passsed by main()
string output_obj_detection_filename = "picture_for_obj_detection.jpg";
stringstream image_stream;
for (const auto &byte : buffer) {
image_stream << byte;
}
curl_mime *mime;
curl_mimepart *part;
mime = curl_mime_init(curl);
part = curl_mime_addpart(mime);
curl_mime_name(part, "min_confidence");
curl_mime_data(part, min_confidence.c_str(), CURL_ZERO_TERMINATED);
part = curl_mime_addpart(mime);
// curl_mime_name(part, "typedfile"); // works for Code Project AI, not for Deepstack
curl_mime_name(part, "image"); // works for both Code Project AI and Deepstack
curl_mime_data(part, image_stream.str().c_str(), buffer.size());
curl_mime_filename(part, output_obj_detection_filename.c_str());
curl_mime_type(part, "image/jpeg");
curl_easy_setopt(curl, CURLOPT_URL, AIserverUrl.c_str());
curl_easy_setopt(curl, CURLOPT_MIMEPOST, mime);
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, write_data);
curl_easy_setopt(curl, CURLOPT_WRITEDATA, &response_data);
res = curl_easy_perform(curl);
char time_now_buf[21];
time_t now;
time(&now);
strftime(time_now_buf, 21, "%Y_%m_%d_%H_%M_%S", localtime(&now));
if (res != CURLE_OK) {
cerr << "Access to AI Object Detection service failed: " << curl_easy_strerror(res) << "\nCheck URL; Is server running?" << endl;
}
} // END if (curl)
curl_easy_cleanup(curl);
// Simulate some time-consuming operation
// sleep(3);
// Write result to shared memory
sem_wait(&sharedMemory2->mutex2);
snprintf(sharedMemory2->message2, BUFFER_SIZE, "%s", response_data.c_str());
sem_post(&sharedMemory2->mutex2);
}
// Function to parse date string
chrono::system_clock::time_point parseDate(const string& dateStr) {
tm tm = {};
istringstream ss(dateStr.substr(22)); // Extract date part from text+date string
// cout << dateStr.substr(22) << "\n\n";
char delim;
ss >> tm.tm_year >> delim >> tm.tm_mon >> delim >> tm.tm_mday >> delim
>> tm.tm_hour >> delim >> tm.tm_min >> delim >> tm.tm_sec;
tm.tm_year -= 1900; // Years since 1900
tm.tm_mon -= 1; // Month index starts from 0
time_t t = mktime(&tm);
return chrono::system_clock::from_time_t(t);
}
int main() {
// Register signal and signal handler. Used to check if CTRL-c has been pressed in console!
signal(SIGINT, signal_callback_handler);
INIReader reader("mycMotDetRecPyC_config.ini");
if (reader.ParseError() < 0) {
cout << "Error: Cannot load config.ini" << endl;
return 1;
}
// CAMERA
// Read variables from the .ini file
// Full rtsp URL of IP camera inluding user name and password when required
string url = reader.Get("camera", "url", "");
// MOTION DETECTION
// Path to the mask file
string mask_path = reader.Get("motion_detection", "mask_path", "");
// Warm up time. The time to be passed after start of program. Hereafter motion dectection will start.
int warmup_time = reader.GetInteger("motion_detection", "warmup_time", 3);
// Simulate a motion Default No. (used for test purposes)
string simulate_a_motion = reader.Get("motion_detection", "simulate_a_motion", "No");
// Show the display window in a resized frame (but not with a mask)
string show_display_window = reader.Get("motion_detection", "show_display_window", "No");
// Show the display window in its original frame (not resized)
string show_display_window_not_resized = reader.Get("motion_detection", "show_display_window_not_resized", "No");
// Show indicator that motion has been detected on display window
string show_motion_detected_msg_on_display_window = reader.Get("motion_detection", "show_motion_detected_msg_on_display_window", "No");
// Show the fps and the date and time on the display window
string show_motion_fps_date_msg_on_display_window = reader.Get("motion_detection", "show_motion_fps_date_msg_on_display_window", "No");
// Show indicator that motion has been detected on display console (terminal window)
string show_motion_detected_msg_on_display_console = reader.Get("motion_detection", "show_motion_detected_msg_on_display_console", "No");
// Show the fps and the date and time on the display console (terminal window)
string show_motion_fps_date_msg_on_display_console = reader.Get("motion_detection", "show_motion_fps_date_msg_on_display_console", "No");
// VIDEO RECORDING
string output_video_path = reader.Get("video_recording", "output_video_path", "./");
// Output prefix video file
string prefix_output_video = reader.Get("video_recording", "prefix_output_video", "Vid_");
// Extension of the video file
string extension_of_video = reader.Get("video_recording", "extension_of_video", ".avi");
// The Frames per second that your camera uses
int fps = reader.GetInteger("video_recording", "fps", 15);
// The codec to be used for writing the videos
string codecString = reader.Get("video_recording", "codec", "XVID");
float maximum_recording_time = reader.GetFloat("video_recording", "maximum_recording_time", 2.5);
// Record duration in seconds
int record_duration = reader.GetInteger("video_recording", "record_duration", 10);
// Buffer x seconds the frames before a (new) motion is detected
int buffer_before_motion = reader.GetInteger("video_recording", "buffer_before_motion", 5);
// Extra time to add to the record duration if new motion is detected
int extra_record_time = reader.GetInteger("video_recording", "extra_record_time", 5);
// Pre-motion recording duration in second. The time before recording should stop wiil be used to detect uf new motion happened.
int before_record_duration_is_passed = reader.GetInteger("video_recording", "before_record_duration_is_passed", 3);
// Output video parameters
string show_timing_for_recording = reader.Get("video_recording", "show_timing_for_recording", "No");
// OBJECT DETECTION
// AI Object Detection Service URL
string AIserverUrl = reader.Get("object_detection", "AIserverUrl", "http://localhost:80/v1/vision/detection");
// Only when an AI object Detection service is installed Object Detection will happen
string AIobject_detection_service = reader.Get("object_detection", "AIobject_detection_service", "No");
// Thresholds for object detection: this is the minimum percentage (fraction) when an object signaled as recognised
string min_confidence = reader.Get("motion_detection", "min_confidence", "0.4");
// Try to find the defined objects and signal them when as recognised
string string_of_objects_for_detection = reader.Get("object_detection", "string_of_objects_for_detection", "person");
// Split the comma-separated string string_of_objects_for_detection into individual values
vector<string> objects_for_detection = splitString(string_of_objects_for_detection, ',');
// Try to find the defined objects and signal them when as recognised
string draw_object_rectangles = reader.Get("object_detection", "draw_object_rectangles", "person");
// Object detection will repeat after x seconds (default 5) when motion has been detected
int object_detection_time = reader.GetInteger("object_detection", "object_detection_time", 3);
// Output picture parameters
string output_obj_picture_path = reader.Get("object_detection", "output_obj_picture_path", "./");
// Output prefix picture file
string prefix_output_picture = reader.Get("object_detection", "prefix_output_picture", "Pic_");
// Show the full json response message from the AI Object Detection Service
string show_AIResponse_message = reader.Get("object_detection", "show_AIResponse_message", "No");
// Show the result(s) from the response message from the AI Object Detection Service
string show_AIObjDetectionResult = reader.Get("object_detection", "show_AIObjDetectionResult", "No"); // When No motion rectangles will be drawn on the screen around the moving objects
// Show the result(s) from the response message from the AI Object Detection Service
string curl_debug_message_on = reader.Get("object_detection", "curl_debug_message_on", "No"); // When No motion rectangles will be drawn on the screen around the moving objects
// PYTHON RUNTIME
// Define the python module name (without .py) extention that handles the ONVIF events messages
string py_moduleName = reader.Get("python_runtime", "moduleName", "myTapoEvents");
// Define the python function name that should be called to execute
string py_functionName = reader.Get("python_runtime", "functionName", "getTapoEventMessage");
// Set the Python runtime configuration MUST be const char* in stead of string
const char* moduleName = py_moduleName.c_str();
const char* functionName = py_functionName.c_str();
// MAIL SMTP SERVER
// Allow sending of emails?
string mail_sendMails = reader.Get("mail_SMTP_server", "sendMails", "No");
// set the host address of the SMTP email server e.g. smtp.gmail.com
string mail_smtpServer = reader.Get("mail_SMTP_server", "smtpServer", "");
// set the port number of the SMTP email server e.g. 587 is common
int mail_smtpPort = reader.GetInteger("mail_SMTP_server", "smtpPort", 587);
// set the emailaddress of username (this is the account) for the SMTP email server
string mail_email_username = reader.Get("mail_SMTP_server", "email_username", "");
// set the email password (this is the account password) for the SMTP email server
string mail_email_password = reader.Get("mail_SMTP_server", "email_password", "");
// set the emailaddress of the sender of the email (complete email address)
string mail_senderEmail = reader.Get("mail_SMTP_server", "senderEmail", "");
// set the emailaddress of the receiver of the email (complete email address)
string mail_recipientEmail = reader.Get("mail_SMTP_server", "recipientEmail", "");
// set the subject for the email (title of the mail)
string mail_email_subject = reader.Get("mail_SMTP_server", "email_subject", "");
// set the body text for the email (the text content of the mail)
string mail_email_body = reader.Get("mail_SMTP_server", "email_body", "This is a test email with an attached file.");
// should a picture of an AI object recognition be attached to the mail?
string mail_email_pictures = reader.Get("mail_SMTP_server", "email_pictures", "Yes");
// should a picture of an AI object recognition be attached to the mail?
string show_email_status_msg_on_display_console = reader.Get("mail_SMTP_server", "show_email_status_msg_on_display_console", "Yes");
// SMTP server configuration
const char* smtpServer = mail_smtpServer.c_str();
const int smtpPort = mail_smtpPort;
const char* username = mail_email_username.c_str();
const char* password = mail_email_password.c_str();
const char* senderEmail = mail_senderEmail.c_str();
const char* recipientEmail = mail_recipientEmail.c_str();
const char* subject = mail_email_subject.c_str();
const char* body = mail_email_body.c_str();
// Create shared memory for getTapoMessages
int shm_fd1 = shm_open("/my_shared_memory1", O_CREAT | O_RDWR, 0666);
ftruncate(shm_fd1, sizeof(SharedMemory1));
SharedMemory1* sharedMemory1 = (SharedMemory1*)mmap(NULL, sizeof(SharedMemory1), PROT_READ | PROT_WRITE, MAP_SHARED, shm_fd1, 0);
sem_init(&sharedMemory1->mutex1, 1, 1); // Initialize semaphore for getTapoMessages
// Create shared memory for postImageAndGetResponse
int shm_fd2 = shm_open("/my_shared_memory2", O_CREAT | O_RDWR, 0666);
ftruncate(shm_fd2, sizeof(SharedMemory2));
SharedMemory2* sharedMemory2 = (SharedMemory2*)mmap(NULL, sizeof(SharedMemory2), PROT_READ | PROT_WRITE, MAP_SHARED, shm_fd2, 0);
sem_init(&sharedMemory2->mutex2, 1, 1); // Initialize semaphore for postImageAndGetResponse
pid_t pid1 = fork();
if (pid1 == 0) {
// Child process (getTapoMessages)
getTapoMessages(sharedMemory1, moduleName, functionName);
return 0; // required!
} else if (pid1 > 0) {
// Parent process (main)
// Do other work while getTapoMessages is running...
// Load the DUMMY mask. The black areas's in the mask are always excluded from Motion detection!
vector<Point> my_poly = {Point(0, 0), Point(2560,0), Point(2560, 1440), Point(0, 1440)};
Mat mask = Mat::zeros(1, 1, CV_8U);
fillPoly(mask, my_poly, Scalar::all(1));
// Invert the mask so that black becomes non-transparent and white becomes transparent
mask = 255 - mask;
// Fork a new process for CURL asyncTask called postImageAndGetResponse
pid_t pid2 = fork();
if (pid2 == 0) {
if (AIobject_detection_service == "Yes") {
postImageAndGetResponse(sharedMemory2, AIserverUrl, min_confidence, mask, show_AIResponse_message, show_AIObjDetectionResult, curl_debug_message_on);
}
return 0; // required!
} else if (pid2 > 0) {
// Parent process (main)
} else {
// Fork for postImageAndGetResponse failed
cerr << "Fork for postImageAndGetResponse failed!" << endl;
return 1;
}
// Initialize camera
VideoCapture cap(url);
if (!cap.isOpened()) {
cout << "Error: Cannot open the camera." << endl;
return -1;
}
cap.set(cv::CAP_PROP_BUFFERSIZE, fps); // do buffer frames
int frame_width = cap.get(cv::CAP_PROP_FRAME_WIDTH);
int frame_height = cap.get(cv::CAP_PROP_FRAME_HEIGHT);
Size frameSize = Size(frame_width, frame_height);
char time_now_buf[21];
time_t now;
time(&now);
strftime(time_now_buf, 21, "%Y_%m_%d_%H_%M_%S", localtime(&now));
cout << "Camera started @ " << time_now_buf << endl;
cout << "Videos are saved @ " << output_video_path << endl;
cout << "Pictures are saved @ " << output_obj_picture_path << endl;
if (AIobject_detection_service == "Yes") {
cout << "Detect Objects, but save ONLY pictures of: " << string_of_objects_for_detection << endl;
}
VideoWriter outputVideo;
// Initialize variables for recording
auto timeCamera = parseDate( "Motion detected: Yes @ 2001-01-01 01:01:01");
auto start_timeCamera = parseDate( "Motion detected: Yes @ 2001-01-01 01:01:01");
int frameCounter = 0;
int teller = 0;
int obj_detection_each_x_frames = fps * object_detection_time;
bool recording_on = false;
int new_record_duration = record_duration;
int bufferSeconds = buffer_before_motion;
// Set buffer size in frames
long unsigned int bufferSize = bufferSeconds * fps;
queue<Mat> frameBuffer;
while (true) {
// Read frame from the camera
Mat frame;
Mat frame_original;
bool motion_detected;
cap.read(frame);
if (frame.empty()) {
break;
}
frame_original = frame.clone();
frameCounter += 1;
string str_frameCounter = to_string(frameCounter);
time(&now);
strftime(time_now_buf, 21, "%Y_%m_%d_%H_%M_%S", localtime(&now));
// Add frame to buffer
frameBuffer.push(frame.clone());
// Maintain buffer size
if (frameBuffer.size() > bufferSize) {
frameBuffer.pop(); // removes first frame from buffer
}
// Read result from shared memory for getTapoMessages
sem_wait(&sharedMemory1->mutex1);
string messages_data(sharedMemory1->message1);
sem_post(&sharedMemory1->mutex1);
if (show_motion_fps_date_msg_on_display_console == "Yes") {
// put frame number and time on screen
// [xK Erases part of the line. If n is 0 (or missing),
// clear from cursor to the end of the line.
// If n is 1, clear from cursor to beginning of the line.
// If n is 2, clear entire line. Cursor position does not change.
time(&now);
strftime(time_now_buf, 21, "%Y_%m_%d_%H_%M_%S", localtime(&now));
cout << "@" << str_frameCounter << " " << time_now_buf << ". Camera: " << messages_data << "\r";
}
if (show_motion_fps_date_msg_on_display_window == "Yes") {
// put frame number and time on screen
time(&now);
strftime(time_now_buf, 21, "%Y_%m_%d_%H_%M_%S", localtime(&now));
putText(frame_original, '@'+str_frameCounter + " " + time_now_buf, Point(10,frame_height - 20), FONT_HERSHEY_SIMPLEX, 0.75, Scalar(0,255,88),2);
}
motion_detected = false;
// Check for motion after a warm up time to avoid initial writing of a file
if (frameCounter >= warmup_time * fps) {
// Print the result for getTapoMessages
// cout << "frameCounter:" << frameCounter << " Result from getTapoMessages asynchronous task: " << messages_data << endl;
if (messages_data.rfind("Motion detected: Yes @", 0) == 0) { // pos=0 limits the search to the prefix
motion_detected = true;
timeCamera = parseDate(messages_data);
}
if (messages_data.rfind("Motion detected: No @", 0) == 0) {
timeCamera = parseDate(messages_data);
}
// Print the result
// cout << "Result from asynchronous task: " << messages_data << endl;
} // END of if (frameCounter >= warmup_time * fps)
// When motion has been detected and displays are required
if (motion_detected) {
if (show_motion_detected_msg_on_display_window == "Yes") {
// cout << "Motion detected" << endl;
// put frame number and time on screen
putText(frame_original, "motion detected", Point(10, frame_height - 40), FONT_HERSHEY_SIMPLEX, 0.75, Scalar(0,255,88),2);
}
if (show_motion_detected_msg_on_display_console == "Yes") {
// cout << "Motion detected" << endl;
// put frame number and time on screen
cout << "motion detected \033[K\r";
}
} // END of if (motion_detected) {
// This simulation is meant for test purposes e.g. to verify the AI Object detection and saving of pictures
if (simulate_a_motion == "Yes") {
if ((frameCounter > 100 && frameCounter < 300) || (frameCounter > 400 && frameCounter < 600))
{motion_detected = true;} else {motion_detected = false;}
}
// If recording is on, then check if extra record time must be added and check if record duration has been passed
if (recording_on) {
// Calculate delta in seconds
chrono::duration<double> delta_rec = timeCamera - start_timeCamera ;
if (motion_detected) {
if (delta_rec.count() >= new_record_duration - before_record_duration_is_passed) {
new_record_duration = new_record_duration + extra_record_time;
}
}
if (delta_rec.count() >= maximum_recording_time * 60) {
new_record_duration = record_duration;
}
if (show_timing_for_recording == "Yes") {
cout << "motion:" <<motion_detected << " timeCamera - start_timeCamera: " << delta_rec.count() << " new_record_duration: " << new_record_duration << " before_record_duration_is_passed: " << before_record_duration_is_passed << " extra time: " << extra_record_time << endl;
}
} // END of First! if (recording_on)
// If motion detected, start recording and set the record duration
if (motion_detected and not recording_on) {
recording_on = true;
start_timeCamera = timeCamera;
time(&now);
strftime(time_now_buf, 21, "%Y_%m_%d_%H_%M_%S", localtime(&now));
cout << "Recording started @ " + string(time_now_buf) << endl;
// set the codec and create VideoWriter object
outputVideo.open(output_video_path + prefix_output_video + time_now_buf + extension_of_video, VideoWriter::fourcc(codecString[0], codecString[1], codecString[2], codecString[3]), fps, frameSize, true);
cout << "To be saved in: " << output_video_path + prefix_output_video + time_now_buf + extension_of_video << endl;
} // END of if (motion_detected and not recording_on)
// If recording is on, check if standard record duration plus applicable extra record time has been passed.
if (recording_on) {
// Check if it's time to stop recording
// Calculate delta in seconds
chrono::duration<double> delta_rec = timeCamera - start_timeCamera;
// cout << delta_rec.count() << " > " << new_record_duration << " ?" << endl;
if (delta_rec.count() > new_record_duration) {
time(&now);
strftime(time_now_buf, 21, "%Y_%m_%d_%H_%M_%S", localtime(&now));
cout << "Recording stopped @ " + string(time_now_buf) << "\033[K\n\n";
recording_on = false;
new_record_duration = record_duration; // reset to start value
// Release the VideoWriter object
if (!outputVideo.isOpened()) {
cerr << "Could not open the output video file for release\n";
return -1;
}
outputVideo.release();
} // END of if chrono::system_clock::now() >= end_time)
else {
// Write frame to the output video
if (!outputVideo.isOpened()) {
cerr << "Could not open the output video file for write\n";
return -1;
}
// Write buffered frames to file
if (!frameBuffer.empty()) {
while (!frameBuffer.empty()) {
teller += 1;
outputVideo.write(frameBuffer.front());
frameBuffer.pop();
}
// cout << frameCounter << " <= Total frame counter | # written frames from buffer => " << teller << endl;
}
else {
outputVideo.write(frame_original);
}
} // END of else branch of if (time(0) >= end_time)
} // END of Second! if (recording_on)
if (recording_on == true and AIobject_detection_service == "Yes") {
// Initialize variables for a new object detection, object_detection_time is defined in config
int modulo_result_AI = frameCounter % obj_detection_each_x_frames;
if (modulo_result_AI == 0) {
// Define vector for the extraction of object detection results
vector<DetectionResult> detectionResults;
// Child process (postImageAndGetResponse)
// Post the image to the AI Object Detection Service and get the detectionResults from the response
postImageAndGetResponse(sharedMemory2, AIserverUrl, min_confidence, frame, show_AIResponse_message, show_AIObjDetectionResult, curl_debug_message_on);
// Read result from shared memory for postImageAndGetResponse
sem_wait(&sharedMemory2->mutex2);
string response_data(sharedMemory2->message2);
sem_post(&sharedMemory2->mutex2);
// check if there is some content in the shared buffer
if (strlen(sharedMemory2->message2) > 0) {
// Print the result for postImageAndGetResponse
// cout << "Result from postImageAndGetResponse asynchronous task: " << response_data << endl;
// Parsing JSON response
Json::Value jsonData;
Json::Value root {};
Json::Reader jsonReader;
if (jsonReader.parse(response_data, jsonData)) {
if (show_AIResponse_message == "Yes") {
time(&now);
strftime(time_now_buf, 21, "%Y_%m_%d_%H_%M_%S", localtime(&now));
cout << time_now_buf << " Successfully parsed JSON data" << endl;
cout << time_now_buf << " JSON data received:" << endl;
cout << time_now_buf << jsonData.toStyledString() << endl;
// const string aiPredictions(jsonData["predictions"].asString());
} // END if (show_AIResponse_message == "Yes")
if (show_AIObjDetectionResult == "Yes") {
const string aiMessage(jsonData["message"].asString());
const string aiSuccess(jsonData["success"].asString());
// Extract object detection results
for (const auto& prediction : jsonData["predictions"]) {
DetectionResult result;
result.label = prediction["label"].asString();
result.confidence = prediction["confidence"].asString();
// prepare confidence fraction (e.g.0.8334) for a percentage (e.g. 83.34%) in cout
float conf;
conf = prediction["confidence"].asFloat() * 100.0;
time(&now);
strftime(time_now_buf, 21, "%Y_%m_%d_%H_%M_%S", localtime(&now));
cout << time_now_buf << " AI Object Detection service message: Success is " << aiSuccess << ". Found: " << result.label << ". Confidence: " << setprecision(3) << conf << "%" << endl;
// Drawing a rectangle in OpenCV with C++ behaves differently than in Python
// In C++ you need Rect((x,y), (x+width_offset, y+height_offset)
result.boundingBox = Rect(prediction["x_min"].asInt(), prediction["y_min"].asInt(),
prediction["x_max"].asInt() - prediction["x_min"].asInt() , prediction["y_max"].asInt() - prediction["y_min"].asInt());
detectionResults.push_back(result);
} // END of for (const auto& prediction : jsonData["predictions"])
} // END of if (show_AIObjDetectionResult == "Yes")
} // END of if (jsonReader.parse(response_data, jsonData))
else
{
time(&now);
strftime(time_now_buf, 21, "%Y_%m_%d_%H_%M_%S", localtime(&now));
cout << time_now_buf << " Could not parse HTTP data as JSON" << endl;
cout << time_now_buf << " HTTP data was:\n" << response_data << endl;
} // END of else branch of if (jsonReader.parse(response_data, jsonData))
} // END of if (strlen(sharedMemory2->message) > 0)
// Clean up
sem_destroy(&sharedMemory1->mutex1);
sem_destroy(&sharedMemory2->mutex2);
shm_unlink("/my_shared_memory1");
shm_unlink("/my_shared_memory2");
// evaluate the detection results of the AI Object Detection service
for (const auto& result : detectionResults) {
// Verify if result.label is equal to one of the values defined in objects_for_detection
// if found then we will draw rectangles and label. In Python it would be: if "car" in ['car', person', 'bicycle']
bool found = false;
for (const string& value : objects_for_detection) {
if (result.label == value) {
found = true;
break;