/*===================================================================== QGroundControl Open Source Ground Control Station (c) 2009 - 2014 QGROUNDCONTROL PROJECT This file is part of the QGROUNDCONTROL project QGROUNDCONTROL 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. QGROUNDCONTROL 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. You should have received a copy of the GNU General Public License along with QGROUNDCONTROL. If not, see . ======================================================================*/ /// @file /// @brief Test for mavlink log collection /// /// @author Don Gagne #include "MavlinkLogTest.h" #include "MainWindow.h" #include "MockLink.h" #include "QGCTemporaryFile.h" #include "QGCApplication.h" #include "UAS.h" #include "MultiVehicleManager.h" UT_REGISTER_TEST(MavlinkLogTest) const char* MavlinkLogTest::_tempLogFileTemplate = "FlightDataXXXXXX"; ///< Template for temporary log file const char* MavlinkLogTest::_logFileExtension = "mavlink"; ///< Extension for log files const char* MavlinkLogTest::_saveLogFilename = "qgroundcontrol.mavlink.ut"; ///< Filename to save log files to MavlinkLogTest::MavlinkLogTest(void) { } void MavlinkLogTest::init(void) { UnitTest::init(); // Make sure temp directory is clear of mavlink logs QDir tmpDir(QStandardPaths::writableLocation(QStandardPaths::TempLocation)); QStringList logFiles(tmpDir.entryList(QStringList(QString("*.%1").arg(_logFileExtension)), QDir::Files)); foreach(QString logFile, logFiles) { bool success = tmpDir.remove(logFile); Q_UNUSED(success); Q_ASSERT(success); } } void MavlinkLogTest::cleanup(void) { // Make sure no left over logs in temp directory QDir tmpDir(QStandardPaths::writableLocation(QStandardPaths::TempLocation)); QStringList logFiles(tmpDir.entryList(QStringList(QString("*.%1").arg(_logFileExtension)), QDir::Files)); QCOMPARE(logFiles.count(), 0); UnitTest::cleanup(); } void MavlinkLogTest::_createTempLogFile(bool zeroLength) { QGCTemporaryFile tempLogFile(QString("%1.%2").arg(_tempLogFileTemplate).arg(_logFileExtension)); tempLogFile.open(); if (!zeroLength) { tempLogFile.write("foo"); } tempLogFile.close(); } void MavlinkLogTest::_bootLogDetectionCancel_test(void) { // Create a fake mavlink log _createTempLogFile(false); // We should get a message box, followed by a getSaveFileName dialog. setExpectedMessageBox(QMessageBox::Ok); setExpectedFileDialog(getSaveFileName, QStringList()); // Kick the protocol to check for lost log files and wait for signals to move through connect(this, &MavlinkLogTest::checkForLostLogFiles, MAVLinkProtocol::instance(), &MAVLinkProtocol::checkForLostLogFiles); emit checkForLostLogFiles(); QTest::qWait(1000); checkExpectedMessageBox(); checkExpectedFileDialog(); } void MavlinkLogTest::_bootLogDetectionSave_test(void) { // Create a fake mavlink log _createTempLogFile(false); // We should get a message box, followed by a getSaveFileName dialog. setExpectedMessageBox(QMessageBox::Ok); QDir logSaveDir(QStandardPaths::writableLocation(QStandardPaths::DocumentsLocation)); QString logSaveFile(logSaveDir.filePath(_saveLogFilename)); setExpectedFileDialog(getSaveFileName, QStringList(logSaveFile)); // Kick the protocol to check for lost log files and wait for signals to move through connect(this, &MavlinkLogTest::checkForLostLogFiles, MAVLinkProtocol::instance(), &MAVLinkProtocol::checkForLostLogFiles); emit checkForLostLogFiles(); QTest::qWait(1000); checkExpectedMessageBox(); checkExpectedFileDialog(); // Make sure the file is there and delete it QCOMPARE(logSaveDir.remove(_saveLogFilename), true); } void MavlinkLogTest::_bootLogDetectionZeroLength_test(void) { // Create a fake empty mavlink log _createTempLogFile(true); // Kick the protocol to check for lost log files and wait for signals to move through connect(this, &MavlinkLogTest::checkForLostLogFiles, MAVLinkProtocol::instance(), &MAVLinkProtocol::checkForLostLogFiles); emit checkForLostLogFiles(); QTest::qWait(1000); // Zero length log files should not generate any additional UI pop-ups. It should just be deleted silently. } void MavlinkLogTest::_connectLogWorker(bool arm) { LinkManager* linkMgr = LinkManager::instance(); Q_CHECK_PTR(linkMgr); MockLink* link = new MockLink(); Q_CHECK_PTR(link); LinkManager::instance()->_addLink(link); linkMgr->connectLink(link); // Wait for the uas to work it's way through the various threads QSignalSpy spyVehicle(MultiVehicleManager::instance(), SIGNAL(activeVehicleChanged(Vehicle*))); QCOMPARE(spyVehicle.wait(5000), true); UAS* uas = MultiVehicleManager::instance()->activeUas(); QVERIFY(uas); QDir logSaveDir; if (arm) { uas->armSystem(); QTest::qWait(1500); // Wait long enough for heartbeat to come through // On Disconnect: We should get a getSaveFileName dialog. logSaveDir.setPath(QStandardPaths::writableLocation(QStandardPaths::DocumentsLocation)); QString logSaveFile(logSaveDir.filePath(_saveLogFilename)); setExpectedFileDialog(getSaveFileName, QStringList(logSaveFile)); } linkMgr->disconnectLink(link); QTest::qWait(1000); // Need to allow signals to move between threads if (arm) { checkExpectedFileDialog(); // Make sure the file is there and delete it QCOMPARE(logSaveDir.remove(_saveLogFilename), true); } } void MavlinkLogTest::_connectLogNoArm_test(void) { _connectLogWorker(false); } void MavlinkLogTest::_connectLogArm_test(void) { //_connectLogWorker(true); } void MavlinkLogTest::_deleteTempLogFiles_test(void) { // Verify that the MAVLinkProtocol::deleteTempLogFiles api works correctly _createTempLogFile(false); MAVLinkProtocol::deleteTempLogFiles(); QDir tmpDir(QStandardPaths::writableLocation(QStandardPaths::TempLocation)); QStringList logFiles(tmpDir.entryList(QStringList(QString("*.%1").arg(_logFileExtension)), QDir::Files)); QCOMPARE(logFiles.count(), 0); }