CreateRecordFileThread.h 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  1. #ifndef _CREATERECORDFILETHREAD_H_
  2. #define _CREATERECORDFILETHREAD_H_
  3. #include "BaseRecordThread.h"
  4. #include "AudioData.h"
  5. #include "GlobalVariable.h"
  6. #include "RingQueue.hpp"
  7. #include "RingQueueManualMutex.hpp"
  8. #include <atomic>
  9. #include <mutex>
  10. #include <QFile>
  11. #include <QDir>
  12. #include <qt5/QtCore/qchar.h>
  13. /**
  14. * @brief 录音状态
  15. *
  16. */
  17. enum class eRecordState
  18. {
  19. eAlarm_Init = 0, /* 初始化状态 */
  20. eAlarm_Recording, /* 录音状态 */
  21. eAlarm_Stopped, /* 停止状态 */
  22. eAlarm_Error, /* 错误状态 */
  23. };
  24. /**
  25. 同一个对比项中多个一致性报警时,主通道的录音可能开始录音时间不同,
  26. 会有多个录音同时存在
  27. */
  28. struct AlarmKey_t
  29. {
  30. int compareItemID; /* 对比项ID */
  31. int roadNum; /* 对比通道编号 */
  32. EAlarmType alarmType; /* 报警类型 */
  33. QDateTime startTime; /* 报警开始时间 */
  34. AlarmKey_t(int id, int num, EAlarmType type, const QDateTime& time)
  35. : compareItemID(id), roadNum(num), alarmType(type), startTime(time) {}
  36. bool operator<(const AlarmKey_t& other) const
  37. {
  38. if(compareItemID != other.compareItemID) {
  39. return compareItemID < other.compareItemID;
  40. }
  41. if(roadNum != other.roadNum) {
  42. return roadNum < other.roadNum;
  43. }
  44. if(alarmType != other.alarmType) {
  45. return alarmType < other.alarmType;
  46. }
  47. return startTime < other.startTime;
  48. }
  49. };
  50. /**
  51. 报警文件队列的值
  52. */
  53. struct AlarmValue_t
  54. {
  55. QString fileName; /* 报警文件名 */
  56. QString fileNameEnd; /* 报警文件结束名,只有在报警结束时才有值 */
  57. eRecordState state; /* 录音状态 */
  58. QDateTime startTime; /* 录音开始时间 */
  59. QDateTime endTime; /* 录音结束时间 */
  60. int alarmStartPos = 0; /* 录音开始位置,单位:秒,报警问价的开始位置不一定是报警的开始位置 */
  61. /* wav文件信息 */
  62. WavHeader wavHeader; /* wav文件头信息 */
  63. int writtenSize = 0; /* 已经写入的大小,单位:字节 */
  64. /* 对比项和通道信息*/
  65. EAlarmType alarmType; /* 报警类型 */
  66. int numConpareItemID; /* 对比项ID */
  67. QString strCompareItemName; /* 对比项名称 */
  68. CompareItemRoadInfo_t itemRoadInfo; /* 对比项通道信息 */
  69. AlarmValue_t() : state(eRecordState::eAlarm_Init) {}
  70. AlarmValue_t(const QString& name, eRecordState s, const QDateTime& start, const QDateTime& end)
  71. : fileName(name), state(s), startTime(start), endTime(end) {}
  72. };
  73. /**
  74. 这个线程类记录一个长的录音文件,1小时一个文件
  75. 功能1:记录一个长的录音文件,1小时一个文件
  76. 1、如果是刚开始录音,到下一个整点记录一个文件,后续都是满1小时一个文件
  77. 2、数据先放到缓冲区中,满1分钟写一次文件
  78. 功能2:
  79. 1、录制报警文件
  80. 2、由其他线程传来报警参数,开启录制,报警结束后,结束录制
  81. 3、由于静音、过载、反相、一致性报警、噪音报警条件是和对比项相关联的,因此这里可能回同时存在多个录音,
  82. 根据对比项ID和对比项通道以及报警类型来区分
  83. 4、报警文件的命名规则:Alarm_CompareItemID_RoadNum_AlarmType_yyyyMMdd_hhmmss-yyyyMMdd_hhmmss.wav
  84. 例如:Alarm_1_1_Silent_20250101_120000-20250101_120100.wav
  85. 文件夹路径格式:
  86. 1、长录音文件夹: Application/ACAServerData/Record/yyyy-MM-dd/SoundCardID-RoadNum/Record_yyyy-MM-dd-hh-mm-ss-yyyy-MM-dd-hh-mm-ss.wav
  87. 例如:Application/2025-01-01/SoundCard1-1/Record_2025-01-01-12-00-00-2025-01-01-12-01-00.wav
  88. 2、报警录音文件夹:Application/ACAServerData/AlarmWav/yyyy-MM-dd/CompareItemID-id/Alarm_RoadNum_AlarmType_yyyyMMdd_hhmmss-yyyyMMdd_hhmmss.wav
  89. 例如:Application/2025-01-01/1-1/Alarm_1_1_Silent_20250101_120000-20250101_120100.wav
  90. */
  91. class CreateRecordFileThread : public BaseRecordThread
  92. {
  93. public:
  94. CreateRecordFileThread(RecordThreadInfo_t& threadInfo);
  95. ~CreateRecordFileThread() override;
  96. /* 设置数据 */
  97. bool setData(const AudioSrcData& srcData) override;
  98. /* 开始录制长文件 */
  99. bool startRecordLongFile(const OneCompareItemRoadInfo_t& compareItemRoadInfo);
  100. /* 停止录制长文件 */
  101. bool stopRecordLongFile(const OneCompareItemRoadInfo_t& compareItemRoadInfo);
  102. /* 开启录制报警信息 */
  103. bool startRecordAlarmFile(const AlarmInfo_t& alarmInfo);
  104. /* 停止录制,alarmInfo既是传入参数,也是传出参数,传出文件路径和开始位置 */
  105. bool stopRecordAlarmFile(AlarmInfo_t& alarmInfo);
  106. private:
  107. /* 执行任务函数 */
  108. void task() override;
  109. /* 初始化一些数据 */
  110. bool initData() override;
  111. /* 清理数据 */
  112. void clearData() override;
  113. /*===============================================================================*/
  114. /* 写入长记录文件 */
  115. bool writeLongRecordFile();
  116. /* 设置今日目录 */
  117. inline bool setTodayPath(bool isNewFile);
  118. /* 打开文件 */
  119. bool openFile(QFile& wavFile, bool isNewFile);
  120. /* 写入音频数据到文件 */
  121. bool writeAudioDataToFile(const AudioSrcData& audioData, const QString& fileName);
  122. /* 生成文件名 */
  123. inline QString generateFileName(const QDateTime& startTime, const QDateTime& endTime) const;
  124. /* 判断是否过了整点 */
  125. inline bool isOneHourPassed();
  126. /*===============================================================================*/
  127. /* 写入报警文件 */
  128. void writeAlarmFile();
  129. /* 创建新的文件 */
  130. void createNewAlarmFile(AlarmValue_t& value, const std::list<AudioSrcData*>& dataList);
  131. /* 生成报警文件名 */
  132. inline QString generateAlarmFileName(const AlarmValue_t& value, bool isNewFile);
  133. /* 设置今日报警文件夹 */
  134. bool setTodayAlarmPath();
  135. /* 根据报警类型的枚举获取字符 */
  136. QString getAlarmTypeString(EAlarmType type) const;
  137. private:
  138. /* 对比项通道信息列表,记录当前对比项的通道信息,如果没有对比项信息就停止录音 */
  139. std::mutex m_mutexCompareItemRoadInfo;
  140. QList<OneCompareItemRoadInfo_t> m_listCompareItemRoadInfo;
  141. std::atomic_bool m_isRequireRecord = false; /* 是否需要录音 */
  142. /* 临时缓存数据, 一分钟写入一次,给这个分配2分钟的数据大小 */
  143. std::mutex m_mutexBuffer;
  144. AudioSrcData m_bufferData; /* 音频数据缓冲区,存储写入的1分钟数据 */
  145. AudioSrcData m_srcData; /* 当前的音频数据,满一分钟后会拷贝到这里,然后写入数据,防止写入的时候阻塞缓冲区 */
  146. int32_t m_writeCriticalSize = 0; /* 写入文件的临界大小,单位:字节,缓存超过这个大小就写入文件 */
  147. int32_t m_oneHourSize = 0; /* 一小时的音频数据大小 */
  148. QDir m_todayDir; /* 今日目录 */
  149. QDate m_todayDateRecord; /* 今日日期,记录长文件用 */
  150. QString m_wavFileName; /* wav文件对象 */
  151. int m_openFileErrorSize = 0; /* 打开文件错误次数 */
  152. WavHeader m_wavHeader; /* 当前wav文件头信息 */
  153. int64_t m_writtenSize = 0; /* 已经写入的数据大小 */
  154. QDateTime m_writtenStartTime; /* 已经写入数据的开始时间点 */
  155. QDateTime m_writtenNowTime; /* 已经写入数据的最后时间点 */
  156. /*===============================================================================*/
  157. /* 报警文件名,key是报警信息,value是文件信息 */
  158. std::mutex m_mutexAlarmFile;
  159. std::map<AlarmKey_t, AlarmValue_t> m_mapAlarmFile;
  160. QDir m_yesterdayDir; /* 昨日目录,用来给还未录制完成的报警文件使用的 */
  161. QDir m_todayDirAlarm; /* 今日报警目录,这个目录只到日期,里面的子文件夹是对比项相关的 */
  162. QDate m_todayDateAlarm; /* 今日日期,记录报警文件用 */
  163. RingQueueManualMutex<AudioSrcData*> m_ringQueue; /* 环形队列,存储报警文件数据 */
  164. QDateTime m_alarmWrittenTime; /* 报警文件中环形队列已写入文件的时间 */
  165. int m_numNewAlarmSeconds = 0; /* 新的报警文件的秒数,新加入的报警文件的秒数,每次写完后清零 */
  166. };
  167. #endif // _CREATERECORDFILETHREAD_H_