Qt編寫物聯網管理平臺43-告警短信轉發

一、前言

系統在運行過程中,會實時採集設備的數據,當採集到的數據發生報警後,可以將報警信息以短信的形式發送給指定的管理員(可以是多個),這樣管理員就算不在現場,也能第一時間知道哪裏發生了報警,可以緊急趕往現場處置。很多時候軟件都是放在現場電腦上運行的,未必現場有人員一直盯着看,通過短信的形式,第一時間送達關鍵人員那裏,及時響應,不錯過任何一次重大報警信息,可以在後臺設定過濾哪些設備產生的報警需要第一時間送達,還可以設置短信發送的間隔,多個接收者。

發送短信在聯網的環境中,可以通過各種互聯網短信服務廠商來接入,在不聯網的環境中,只有一個辦法那就是通過插卡的短信設備來發送,有標準的AT協議,基本上硬件都提供了串口接口,所以程序上其實就是串口通信,通過收發協議數據來解析,比如發送 AT+CSCA? 命令表示查詢短信中心號碼,AT+CMGR=1 表示讀取序號=1的短信內容。爲了能夠支持各個廠家的設備,特意封裝的通用的短信類,支持長短信發送,支持多個收件人,多線程發送。

二、功能特點

2.1 軟件模塊

  1. 設備監控模塊,包括數據監控(表格形式展示)、設備面板(面板形式展示)、地圖監控(地圖形式展示)、曲線監控(曲線形式展示)。
  2. 數據查詢模塊,包括報警記錄、運行記錄、操作記錄。
  3. 系統設置模塊,包括基本設置、端口管理、控制器管理、探測器管理、報警聯動、類型設置等。
  4. 其他設置模塊,包括用戶管理、地圖管理、位置調整、組態設計、設備調試等。

2.2 基礎功能

  1. 設備數據採集,支持串口、網絡,串口可設置串口號、波特率,網絡可設置IP地址、通訊端口。
  2. 每個端口支持採集週期時間,默認1秒鐘一個設備。
  3. 支持設置通訊超時次數,默認3次。
  4. 支持最大重連時間,用於重新讀取離線的設備。
  5. 控制器信息,能夠添加控制器名稱,選擇控制器地址、控制器型號,設置該控制器下面的探測器數量。
  6. 探測器信息,能夠添加位號、探測器型號、氣體種類、氣體符號、高報值、低報值、緩衝值、清零值、是否啓用、報警聲音、背景地圖、存儲週期、數值換算小數點位數、報警延時時間、報警的類型(HH,LL,HL)等。
  7. 類型管理可配置控制器型號、探測器型號、氣體種類、氣體符號等。
  8. 地圖支持導入和刪除,所有的探測器在地圖上的位置可自由拖動保存。
  9. 端口信息、控制器信息、探測器信息、類型信息、用戶信息等,都支持導入、導出、導出到excel、打印。
  10. 運行記錄、報警記錄、操作記錄,都支持多條件組合查詢,比如時間段、控制器、探測器等,所有記錄支持導出到excel/pdf和打印。
  11. 運行記錄、報警記錄、操作記錄都可刪除指定時間範圍內的數據。
  12. 系統設置可選擇對應表最大保存記錄數,自動清理早期數據,留出足夠的空間存儲重要的數據。
  13. 報警短信轉發,支持多個接收手機號碼,可設定發送間隔,比如即時發送或者6個小時發送一次所有的報警信息,短信內容過長,自動拆分多條短信。
  14. 報警郵件轉發,支持多個接收郵箱,可設定發送間隔,比如即時發送或者6個小時發送一次所有的報警信息,支持附件發送。
  15. 設置軟件的中文標題、英文標題、logo路徑、版權所有等。
  16. 開關設置開機運行、報警聲音、自動登錄、記住密碼等。
  17. 報警聲音可設置播放次數,界面風格樣式提供18套皮膚文件選擇。
  18. 用戶管理,包括用戶權限配置,不同用戶可以有不同模塊的權限。
  19. 用戶登錄和用戶退出,可以記住密碼和自動登錄,超過三次報錯提示並關閉程序。
  20. 四種監控模式,設備面板監控、地圖監控、表格數據監控、曲線數據監控,可自由切換,四種模式下都實時展示採集到的數據,報警閃爍等。
  21. 報警繼電器聯動,一個位號可以跨串口聯動多個模塊和繼電器號,支持多對多。

2.3 特色功能

  1. 通信協議支持modbus_com、modbus_tcp_rtu,後期拓展mqtt等協議。
  2. 數據源除了真實的硬件設備採集,還可選數據庫採集,這樣用戶可以安排其他程序員比如java程序員將前端採集好的數據放到數據庫,本系統直接從數據庫採集即可。數據庫採集模式可以作爲通用的系統使用,更適合多人多系統協作。
  3. 智能跳過超時的設備,加快對在線設備的採集速度,當設備數量很多的時候尤其有用。
  4. 對智能跳過的超時的設備,在設定的重連時間自動採集一次,以便探測設備是否又重新上線。
  5. 每個探測器可控是否啓用,不啓用則不會採集,也不會在界面顯示,相當於運行階段臨時關閉。
  6. 探測器可設置緩衝值和報警延時時間,在該值附近波動產生的報警,不計入報警,只有持續處於報警值且超過報警延時時間纔算真正報警,這樣可以規避很多波動導致的誤報。
  7. 探測器可設置存儲週期,按照設定的時間來存儲一條運行記錄,可以按照重要程度對重要性高的設定存儲週期短一些,不重要的設定大一些,這樣可以節省不少的存儲空間,也保證了重要的數據及時存儲。
  8. 探測器可設置清零值,在一些高精度高靈敏的設備可能出廠的時候默認值未必是0,需要設定清零值來表示初始值。
  9. 探測器可設置小數點,用於計算後的真實數據控制小數點點位顯示,相當於除以10、除以100、除以1000,這樣大部分的探測器數據直接通過小數點位設置控制真實換算後的值,極個別的需要特殊轉換的可以在通信協議中約定。
  10. 探測器報警的類型支持多種,有些設備是高於某個值高報,低於某個值低報,而有些設備是在最小值最大值範圍內是高報,低於最小值低報,高於最大值正常。這樣可以分情況處理,涵蓋各種報警類型。
  11. 原創數據導入、導出、打印機制,跨平臺不依賴任何組件,瞬間導出數據。
  12. 導出到excel的記錄支持所有excel、wps等表格文件版本,不依賴excel等軟件。
  13. 高報顏色、低報顏色、正常顏色、默認值顏色等,都可以自由設置。
  14. 支持雲端數據同步,將本地採集到的數據實時同步到雲端。
  15. 支持網絡轉發和網絡接收,網絡接收開啓後,軟件從udp接收數據進行解析。網絡轉發支持多個目標IP,這樣就實現了本地採集的軟件,自由將數據轉到客戶端,隨時查看採集到的數據。
  16. 自動記住用戶最後停留的界面以及其他配置信息,重啓後自動應用。
  17. 報警自動切換到對應的地圖,探測器按鈕閃爍,表格數據對應顏色顯示。
  18. 雙擊探測器圖標,彈出對應探測器詳細信息,可以根據需要定製回控操作。
  19. 數據庫支持多種,包括sqlite、mysql、sqlserver、postgresql、oracle、人大金倉等。
  20. 本地設備採集到的數據實時上傳到雲端,以便手機APP或者web等其他方式提取。
  21. 自帶設備模擬工具,支持不同型號的多個設備數據模擬,同時還帶數據庫數據模擬,以便在沒有設備的時候測試數據。
  22. 標準modbus協議,各種控制器類型、探測器類型、種類、符號等全部自定義,非常靈活和強大,通信協議示例數據非常完整,通用各種modbus協議系統,適用於各種應用場景接入。
  23. 同時集成了串口通信、網絡通信、數據庫通信、數據導入導出打印、通信協議解析、界面UI、全局換膚等衆多組件和知識點,非常適合新手入門和進階。
  24. 支持xp、win7、win10、、win11、linux、mac、各種國產系統(UOS、中標麒麟、銀河麒麟等)、嵌入式linux等系統。
  25. 註釋完整,項目結構清晰,超級詳細完整的使用開發手冊,精確到每個代碼文件的功能說明,不斷持續迭代版本。

三、體驗地址

  1. 國內站點:https://gitee.com/feiyangqingyun
  2. 國際站點:https://github.com/feiyangqingyun
  3. 個人主頁:https://blog.csdn.net/feiyangqingyun
  4. 知乎主頁:https://www.zhihu.com/people/feiyangqingyun
  5. 產品主頁:https://blog.csdn.net/feiyangqingyun/article/details/97565652
  6. 在線文檔:https://feiyangqingyun.gitee.io/qwidgetdemo/iotsystem/
  7. 體驗地址:https://pan.baidu.com/s/1ZxG-oyUKe286LPMPxOrO2A 提取碼:o05q 文件名:bin_iotsystem.zip。
  8. 文章導航:https://qtchina.blog.csdn.net/article/details/121330922

四、效果圖



五、相關代碼

void SendMsgThread::readMsg()
{
    isRead = false;
    QString data = com->readAll();
    data.replace(" ", "");
    data.replace("\r", "");
    data.replace("\n", "");
    if (data.length() <= 0) {
        return;
    }

    //從SIM卡上讀取短信回覆  "AT+CMGR=5+CMGR:"RECREAD","+8618001797656",,"16/05/28,18:12:52+32"006:WD+05.1,ZL139.6,YL1.2,AUTO,YSJ-ONOK"
    //主動收到短信          "AT+CMGR=6+CMGR:"RECUNREAD","+8618001797656",,"16/05/28,18:20:51+32"006:WD+05.1,ZL139.6,YL1.2,AUTO,YSJ-ONOK"
    qDebug() << TIMEMS << "Receive:" << data;
    emit receiveData(data);

    if (data.endsWith(",,0OK")) {
        clear();
        emit readMsgFinish();
        return;
    }

    //收到行短信後會收到 +CMTI:"SM",2 ,其中2爲短信索引
    //接着根據新短信的索引讀取短信,讀取短信成功返回 AT+CMGR=5+CMGR:0,,230891683110304105F0240D91688110978310F000004121519115322303E13919
    if (data.contains("CMTI")) {
        //獲取新短信的索引號
        msgIndex = data.split(",").at(1).toInt();
        writeData(QString("AT+CMGR=%1\r").arg(msgIndex));
    } else if (data.contains("CMGR")) {
        //一步步解析服務中心號碼,發件人號碼,發送時間,短信內容
        QStringList list = data.split(",");
        if (list.count() < 3) {
            return;
        }

        if (mode == 0) {
            QString unicode = list.at(2);
            int len = unicode.length();

            //去除末尾ok兩個字符
            unicode = unicode.mid(0, len - 2);
            if (unicode == "0") {
                clear();
                emit readMsgFinish();
                return;
            }

            int index = unicode.indexOf("08");
            QString centerNum = unicodeToTel(unicode.mid(index + 6, 12));
            QString msgTel = unicodeToTel(unicode.mid(index + 26, 12));
            QString msgType = unicode.mid(index + 40, 2);
            QString msgTime = unicodeToTime(unicode.mid(index + 42, 12));
            QString msgContent = unicode.mid(index + 58);

            //類型爲 00 則採用的是 7bit 編碼,需要用 7bit 解碼
            //類型爲 08 則採用的是 unicode 編碼
            if (msgType == "00") {
                msgContent = unicodeToMsg7Bit(msgContent);
            } else if (msgType == "08") {
                msgContent = unicodeToMsg(msgContent);
            }

            //發送收到新短信信號
            emit receiveMsg(centerNum, msgTel, msgTime, msgContent);
        } else {
            QString centerNum = "13800000000";
            QString msgTel = list.at(1).mid(4, 11);
            QString msgTime = list.at(3);
            msgTime = msgTime.mid(1, msgTime.length());
            msgTime.replace("/", "-");
            msgTime = "20" + msgTime + " " + list.at(4).mid(0, 8);

            QStringList str;
            for (int i = 4; i < list.count(); ++i) {
                str.append(list.at(i));
            }

            QString msgContent = str.join(",");
            msgContent = msgContent.split("\"").at(1);
            msgContent = msgContent.mid(0, msgContent.length() - 2);

            //發送收到新短信信號
            emit receiveMsg(centerNum, msgTel, msgTime, msgContent);
        }
    } else if (data.contains("NOCARRIER")) {
        emit callFinsh();
    } else if (data.contains("NOANSWER")) {
        emit callError();
    } else if (data.contains("RING")) {
        QString tel;
        QStringList list = data.split("\"");
        if (list.count() > 1) {
            tel = list.at(1);
        }
        emit receiveCall(tel);
    } else if (data.contains("DTMF")) {
        int index = data.indexOf("DTMF");
        QString key = data.mid(index + 5, 1);
        emit receiveKey(key);
    }
}

void SendMsgThread::deleteMsg()
{
    //自動刪除SIM卡上的短信
    if (autoDelete) {
        deleteMsg(msgIndex);
    }
}

QString SendMsgThread::sendAT(const QString &cmd, int sleep)
{
    if (!isOpen || !isLive) {
        return "";
    }

    //主動執行命令期間,斷開收到數據信號槽連接
    disconnect(com, SIGNAL(readyRead()), this, SLOT(readData()));
    writeData(QString("%1\r").arg(cmd));
    msleep(sleep);

    QString data = com->readAll();
    data.replace(" ", "");
    data.replace("\r", "");
    data.replace("\n", "");
    if (data.length() > 0) {
        qDebug() << TIMEMS << "Result:" << data;
        emit receiveData(data);
    }

    //完成處理後,重新綁定收到數據信號槽連接
    connect(com, SIGNAL(readyRead()), this, SLOT(readData()), Qt::DirectConnection);
    return data;
}

bool SendMsgThread::sendMsg(const QString &tel, const QString &msg)
{
    bool ok = false;
    char ctrlz  = 26;
    int sleepTime;
    QString msgContent;
    QString sendMsg;

    if (mode == 0) {
        sleepTime = 6000;
        msgContent = QString("%1%2%3%4%5").arg("0011000D91").arg(getMsgTel(tel))
                     .arg("000801").arg(getMsgHex(msg)).arg(ctrlz);
        sendMsg = QString::number(getMsgLen(msg));
    } else {
        sleepTime = 1000;
        msgContent = QString("%1%2").arg(msg).arg(ctrlz);
        sendMsg = tel;
    }

    //首先進入發送短信狀態
    QString result = sendAT(QString("AT+CMGS=%1\r").arg(sendMsg), 300);
    //返回結果最末一位爲 > 後輸入短信內容編碼,發送短信正文
    if (result.right(1) == ">") {
        result = sendAT(msgContent, sleepTime);
        if (mode == 0) {
            if (result.right(2) == "OK") {
                ok = true;
            }
        } else {
            if (result.right(2) == "OK" || result.startsWith(msg)) {
                ok = true;
            }
        }
    }

    return ok;
}

void SendMsgThread::startCall(const QString &tel)
{
    sendAT(QString("ATD%1;").arg(tel), 100);
}

void SendMsgThread::stopCall()
{
    sendAT(QString("ATH"), 100);
}

void SendMsgThread::answerCall()
{
    sendAT(QString("ATA"), 100);
}
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章