Compare commits
	
		
			207 Commits
		
	
	
		
			v1.3.17
			...
			1acec4d876
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 1acec4d876 | |||
| 46c481b862 | |||
| 8945ead5d6 | |||
| bc064c38c6 | |||
| f741fb96f9 | |||
| 8d1c719626 | |||
| b5389a599e | |||
| ea2f42beed | |||
| 22e751e197 | |||
| 05c7d11ef8 | |||
| 63dc70dda9 | |||
| e5ab9b98cd | |||
| 4477647acf | |||
| d64b029a2d | |||
| e24022a612 | |||
| 60d78790ba | |||
| 543e712c62 | |||
| 8402fde6b5 | |||
| f15cc636c6 | |||
| adaa12395e | |||
| 853c29e448 | |||
| 78c214ceb4 | |||
| 88bec387c5 | |||
| 05b68f21c2 | |||
| 9ca394db46 | |||
| 0d268a1d9d | |||
| da102b5ed0 | |||
| 7031c0e349 | |||
| c528ce377d | |||
| 5e7c848fca | |||
| 3165d77f49 | |||
| f0d2d5f958 | |||
| 1da4c3b224 | |||
| f1cedbf1d5 | |||
| 0933274c82 | |||
| 2aa676f18c | |||
| 6c0d49b90c | |||
| 18d04d15cf | |||
| b81120f8dc | |||
| 8b2fcb25db | |||
| 698cf74516 | |||
| 20681b0d6c | |||
| 763647c145 | |||
| 4aad14b181 | |||
| 0a03228dea | |||
| 23551066c1 | |||
| 24f2ba7c44 | |||
| 2ed6768953 | |||
| a1327388bc | |||
| 8d18ae10fe | |||
| 31bc2d0fa2 | |||
| 175b8fd3a1 | |||
| a57fa6c31e | |||
| c22c924e38 | |||
| bdc64920a2 | |||
| 7bc9fa0c04 | |||
| 334da1fe4a | |||
| e5054582c2 | |||
| bbd287c92b | |||
| 5c152c9dc2 | |||
| 367ffaa3a1 | |||
| 2fd004f249 | |||
| 721a156fce | |||
| b7576d04e1 | |||
| 753954ecd7 | |||
| df7cabe2c1 | |||
| cfca8d2ef4 | |||
| 829932f29d | |||
| 8968667c23 | |||
| a5e0154757 | |||
| f346390631 | |||
| c77fa097c1 | |||
| 074e60d8dc | |||
| 8eff6e5d10 | |||
| 4233ca8637 | |||
| 68fc83ba67 | |||
| 052028afe8 | |||
| a240711946 | |||
| 75fdca95d9 | |||
| 9b087f62f9 | |||
| 7d38cc1269 | |||
| f5baad16d5 | |||
| b35e6812aa | |||
| d7e709d997 | |||
| 4d93aa1392 | |||
| b0f1e0a493 | |||
| 0dffc1d1c2 | |||
| 3581dd4b1d | |||
| 8d2d7bbb15 | |||
| e2e7c0772d | |||
| 807cea1042 | |||
| 80712964ef | |||
| 978e98b2ee | |||
| 44c2820130 | |||
| 123f22cd21 | |||
| b60af1a4fc | |||
| bb3ecc4b76 | |||
| fa30fe045b | |||
| bae014822d | |||
| 6c472b542e | |||
| 7578746d2f | |||
| 0b1ed62df1 | |||
| 76ec41c291 | |||
| 630cd36f13 | |||
| cbe8bb7aeb | |||
| e04636e3f7 | |||
| 728185ddb9 | |||
| aaa485e2fc | |||
| 9a9480035b | |||
| 0a43654f9b | |||
| 4dba36a420 | |||
| f9ce6a6c1b | |||
| c2c6bc3f8a | |||
| 8c02ebcf15 | |||
| 100bfd63ab | |||
| 23ff4977d9 | |||
| 8da6443833 | |||
| 6b3ebde2b5 | |||
| 41392a98e3 | |||
| 1c9bb11f0b | |||
| ec0e687c21 | |||
| e040e784fc | |||
| d2300b87c6 | |||
| 0dff2ece75 | |||
| db24853062 | |||
| 530ea33460 | |||
| 005e4d249f | |||
| 1c0786e28c | |||
| 0aa8d9ba5a | |||
| 1ecb844b64 | |||
| 9a687e6628 | |||
| 979afa37d3 | |||
| 1e1820724d | |||
| d90954c6eb | |||
| cf77d0ff76 | |||
| 73d02d214a | |||
| 6a67d8e9b0 | |||
| 30d8cc3684 | |||
| bc9ebb7d68 | |||
| e6f6d43bf2 | |||
| 854c8b9706 | |||
| d521fd977a | |||
| ebbdc2f864 | |||
| 62496c5d95 | |||
| edd606fe78 | |||
| 1748c35c45 | |||
| 4c46932a3c | |||
| 686e113c3a | |||
| 6ddfbbfd9e | |||
| 7bfbdc0f07 | |||
| dd591fdd23 | |||
| 18c7b656c3 | |||
| 5e9b05e887 | |||
| 37aae73f21 | |||
| 0b4eed9dc0 | |||
| 12f48ad1bb | |||
| 5d7d1a2870 | |||
| 5d7f13a254 | |||
| d332a990d5 | |||
| bfa39eb3df | |||
| d7fcfa3d0f | |||
| f1e449c108 | |||
| 32346c2665 | |||
| 38e7bf4985 | |||
| 6df73e1082 | |||
| 19250a0a2f | |||
| 6f5c8103e4 | |||
| 07dcf0ba30 | |||
| 1e379cf086 | |||
| e87456f26b | |||
| 0c9a7bc7b5 | |||
| a35c6afcb8 | |||
| b06f4f46bb | |||
| 33d45eab2e | |||
| d2e11d8f07 | |||
| fbffdc923f | |||
| 8faf5af2bf | |||
| 1d532c13a4 | |||
| d2d3afc28e | |||
| f4bb201633 | |||
| b68cf1fd1d | |||
| 0888c1b525 | |||
| a6faf9ce21 | |||
| 5577cc5d6d | |||
| 9967d5cc45 | |||
| e20ed57bd5 | |||
| 6986007b16 | |||
| 9ecdf73bc0 | |||
| 59d6c49cb2 | |||
| f6a8059e2d | |||
| 4372cb578b | |||
| d683a8fc32 | |||
| 4e92522578 | |||
| b45f3a04b4 | |||
| 0db39746db | |||
| f2844aa4d9 | |||
| d4043bd7d2 | |||
| 89b639c0ed | |||
| 1ad13d9a8a | |||
| 6f2cbb0a26 | |||
| c15cebf503 | |||
| 5ee1308c9d | |||
| 746d96ca7c | |||
| f387eaedea | |||
| 0fd977c399 | |||
| d53d72c536 | |||
| f4be1f3f51 | 
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| tags | ||||
| *.tags | ||||
| *.user | ||||
| @@ -14,3 +14,5 @@ dry-run=false | ||||
| extended-version=false | ||||
| yocto-version=false | ||||
| yocto-install=false | ||||
| always-download-config=true | ||||
| always-download-dc=false | ||||
|   | ||||
							
								
								
									
										3
									
								
								ATBUpdateTool.pro
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								ATBUpdateTool.pro
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| TEMPLATE = subdirs | ||||
| CONFIG += ordered | ||||
| SUBDIRS = DownloadDCFirmware DownloadDCJsonFiles UpdatePTUDevCtrl | ||||
							
								
								
									
										0
									
								
								DownloadDCFirmware/ATBDownloadDCFirmware.ini
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								DownloadDCFirmware/ATBDownloadDCFirmware.ini
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										96
									
								
								DownloadDCFirmware/DownloadDCFirmware.pro
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										96
									
								
								DownloadDCFirmware/DownloadDCFirmware.pro
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,96 @@ | ||||
| QT += core gui | ||||
| QT += widgets serialport network | ||||
|  | ||||
| TARGET = ATBDownloadDCFirmware | ||||
|  | ||||
| VERSION="0.1.0" | ||||
| win32 { | ||||
|     BUILD_DATE=$$system("date /t") | ||||
|     BUILD_TIME=$$system("time /t") | ||||
| } else { | ||||
|     BUILD_DATE=$$system("date +%d-%m-%y") | ||||
|     BUILD_TIME=$$system("date +%H:%M:%S") | ||||
| } | ||||
|  | ||||
| GIT_COMMIT=$$system("git log -1 --format=oneline | cut -d' ' -f1") | ||||
|  | ||||
| EXTENDED_VERSION="$${VERSION}-$${GIT_COMMIT}" | ||||
|  | ||||
| CONFIG += c++17 | ||||
|  | ||||
| DEFINES+=APP_VERSION=\\\"$$VERSION\\\" | ||||
| DEFINES+=APP_BUILD_DATE=\\\"$$BUILD_DATE\\\" | ||||
| DEFINES+=APP_BUILD_TIME=\\\"$$BUILD_TIME\\\" | ||||
| DEFINES+=APP_EXTENDED_VERSION=\\\"$$EXTENDED_VERSION\\\" | ||||
|  | ||||
| # keep comments, as /* fall through */ | ||||
| QMAKE_CXXFLAGS += -C | ||||
| QMAKE_CXXFLAGS += -g | ||||
| QMAKE_CXXFLAGS += -Wno-deprecated-copy -O | ||||
|  | ||||
| contains( CONFIG, PTU5 ) { | ||||
|     greaterThan(QT_MAJOR_VERSION, 4): QT += serialport | ||||
|     CONFIG += link_pkgconfig | ||||
|     lessThan(QT_MAJOR_VERSION, 5):   PKGCONFIG += qextserialport | ||||
|     QMAKE_CXXFLAGS += -O2 -std=c++17   # for GCC >= 4.7 | ||||
|     # QMAKE_CXXFLAGS += -Wno-deprecated-copy | ||||
|     PTU5BASEPATH = /opt/devel/ptu5 | ||||
|  | ||||
|     INCLUDEPATH += $$PTU5BASEPATH/qt/libs/devicecontroller/include | ||||
|     LIBS += -L$$PTU5BASEPATH/qt/libs/devicecontroller/library | ||||
|  | ||||
|     ARCH = PTU5 | ||||
|     DEFINES+=PTU5 | ||||
| } | ||||
| contains( CONFIG, PTU5_YOCTO ) { | ||||
|     greaterThan(QT_MAJOR_VERSION, 4): QT += serialport | ||||
|     QMAKE_CXXFLAGS += -std=c++17   # for GCC >= 4.7 | ||||
|     # QMAKE_CXXFLAGS += -Wno-deprecated-copy | ||||
|     PTU5BASEPATH = /opt/devel/ptu5 | ||||
|     ARCH = PTU5 | ||||
|     DEFINES+=PTU5 | ||||
|  | ||||
|     # add qmqtt lib | ||||
|     #LIBS += -lQt5Qmqtt | ||||
| } | ||||
| contains( CONFIG, DesktopLinux ) { | ||||
|     greaterThan(QT_MAJOR_VERSION, 4): QT += serialport | ||||
|     lessThan(QT_MAJOR_VERSION, 5):    CONFIG += extserialport | ||||
|     # QMAKE_CC = ccache $$QMAKE_CC | ||||
|     # QMAKE_CXX = ccache $$QMAKE_CXX | ||||
|     QMAKE_CXXFLAGS += -std=c++17 | ||||
|     # QMAKE_CXXFLAGS += -Wno-deprecated-copy | ||||
|     linux-clang {  QMAKE_CXXFLAGS += -Qunused-arguments   } | ||||
|     ARCH = DesktopLinux | ||||
|     DEFINES+=DesktopLinux | ||||
| } | ||||
|  | ||||
| SOURCES += \ | ||||
|         main.cpp \ | ||||
|         mainwindow.cpp \ | ||||
|         sender_thread.cpp \ | ||||
|         receiver_thread.cpp \ | ||||
|         worker_thread.cpp \ | ||||
|         ../common/src/message_handler.cpp | ||||
|  | ||||
| HEADERS += \ | ||||
|         mainwindow.h \ | ||||
|         sender_thread.h \ | ||||
|         receiver_thread.h \ | ||||
|         worker_thread.h \ | ||||
|         ../common/include/message_handler.h | ||||
|  | ||||
| OTHER_FILES += \ | ||||
|     ATBDownloadDCFirmware.ini | ||||
|  | ||||
|  | ||||
| FORMS += \ | ||||
|     mainwindow.ui | ||||
|  | ||||
| ########################################################################################## | ||||
| # for running program on target through QtCreator | ||||
| contains( CONFIG, PTU5 ) { | ||||
|    qnx: target.path = /tmp/$${TARGET}/bin | ||||
|    else: unix:!android: target.path = /opt/app/tools/atbupdate/ | ||||
|    !isEmpty(target.path): INSTALLS += target | ||||
| } | ||||
							
								
								
									
										0
									
								
								DownloadDCFirmware/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								DownloadDCFirmware/README.md
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										53
									
								
								DownloadDCFirmware/main.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								DownloadDCFirmware/main.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,53 @@ | ||||
| #include <QtGlobal> | ||||
| #include <QCoreApplication> | ||||
| #include <QApplication> | ||||
| #include <QDebug> | ||||
| #include <QTimer> | ||||
| #include <QFileInfo> | ||||
| #include <QThread> | ||||
|  | ||||
| #include "../common/include/message_handler.h" | ||||
| #include "mainwindow.h" | ||||
|  | ||||
| #ifdef __linux__ | ||||
| #include <sys/sysinfo.h> | ||||
| #endif | ||||
|  | ||||
| #ifdef PTU5 | ||||
| #define SERIAL_PORT "ttymxc2" | ||||
| #else | ||||
| #define SERIAL_PORT "ttyUSB0" | ||||
| #endif | ||||
|  | ||||
| int main(int argc, char **argv) { | ||||
|     QByteArray const value = qgetenv("LC_ALL"); | ||||
|     if (value != "C") { | ||||
|         qputenv("LC_ALL", "C"); | ||||
|     } | ||||
|     // qputenv("XDG_RUNTIME_DIR", "/var/run/user/0"); | ||||
|  | ||||
|     openlog("ATB-UPDATE-DC-FIRMWARE", LOG_PERROR | LOG_PID | LOG_CONS, LOG_USER); | ||||
|  | ||||
|     QApplication a(argc, argv); | ||||
|     QApplication::setApplicationName("ATBUpdateTool"); | ||||
|     QApplication::setApplicationVersion(APP_VERSION); | ||||
|  | ||||
|     if (!messageHandlerInstalled()) { // change internal qt-QDebug-handling | ||||
|         atbInstallMessageHandler(atbDebugOutput); | ||||
|         setDebugLevel(LOG_NOTICE); | ||||
|     } | ||||
|  | ||||
|     QThread::currentThread()->setObjectName("main thread"); | ||||
|     qInfo() << "Main thread" << QThread::currentThreadId(); | ||||
|  | ||||
|     MainWindow mw; | ||||
|  | ||||
|     mw.setWindowFlags(Qt::Window | Qt::FramelessWindowHint); | ||||
|     //mw.showFullScreen(); | ||||
|  | ||||
|     qCritical() << "SHOW"; | ||||
|  | ||||
|     mw.show(); | ||||
|  | ||||
|     return a.exec(); | ||||
| } | ||||
							
								
								
									
										347
									
								
								DownloadDCFirmware/mainwindow.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										347
									
								
								DownloadDCFirmware/mainwindow.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,347 @@ | ||||
| #include "mainwindow.h" | ||||
| #include "ui_mainwindow.h" | ||||
|  | ||||
| #include <QDateTime> | ||||
| #include <QMessageBox> | ||||
| #include <QDebug> | ||||
| #include <QScrollBar> | ||||
| #include <QEvent> | ||||
|  | ||||
|  | ||||
| MainWindow::MainWindow(QWidget *parent) | ||||
|     : QMainWindow(parent) | ||||
|     , ui(new Ui::MainWindow) | ||||
|     , m_serial(new QSerialPort(this)) | ||||
|     , m_width(70) { | ||||
|  | ||||
|     this->setStatusBar(new QStatusBar(this)); | ||||
|     QFont f; | ||||
|     f.setStyleHint(QFont::Monospace); | ||||
|     f.setWeight(QFont::Bold); | ||||
|     f.setFamily("Misc Fixed"); | ||||
|     f.setPixelSize(12); | ||||
|     this->statusBar()->setFont(f); | ||||
|  | ||||
|     ui->setupUi(this); | ||||
|  | ||||
|     openSerialPort(); | ||||
|  | ||||
|     //ui->updateProgress->setRange(0, 100); | ||||
|     //ui->updateProgress->reset(); | ||||
|  | ||||
| #if 0 | ||||
|     QStringList lst; | ||||
|     QString start = QDateTime::currentDateTime().toString(Qt::ISODate); | ||||
|     lst << QString("Start: ") + start.leftJustified(m_width-10); | ||||
|     lst << QString("").leftJustified(m_width-3, '='); | ||||
|     lst << QString("Update tool version: %1 - %2 %3").arg(APP_VERSION).arg(APP_BUILD_DATE).arg(APP_BUILD_TIME).leftJustified(m_width-3); | ||||
|     lst << QString("Machine number     : %1 ").arg(m_worker->machineNr()).leftJustified(m_width-3); | ||||
|     lst << QString("Customer number    : %1 ").arg(m_worker->customerNr()).leftJustified(m_width-3); | ||||
|     lst << QString("Zone number        : %1 (%2)").arg(m_worker->zoneNr()).arg(Utils::zoneName(m_worker->zoneNr())).leftJustified(m_width-3); | ||||
|     lst << QString("APISM version      : %1").arg(m_worker->apismVersion()).leftJustified(m_width-3); | ||||
|     lst << QString("").leftJustified(m_width-3, '='); | ||||
|  | ||||
|     ui->updateStatus->setText(lst.join('\n')); | ||||
|     ui->updateStatus->setEnabled(true); | ||||
|     // ui->updateStatus->installEventFilter(this); | ||||
|  | ||||
|     m_startTimer = new QTimer(this); | ||||
|     connect(m_startTimer, SIGNAL(timeout()), m_worker, SLOT(start())); | ||||
|     m_startTimer->setSingleShot(true); | ||||
|     m_startTimer->start(1000); | ||||
|  | ||||
|     m_exitTimer = new QTimer(this); | ||||
|     connect(m_exitTimer, SIGNAL(timeout()), ui->exit, SLOT(click())); | ||||
|     m_exitTimer->setSingleShot(true); | ||||
|     m_exitTimer->start(1800 * 1000); | ||||
|  | ||||
|     connect(ui->exit, SIGNAL(clicked()),this,SLOT(onQuit())); | ||||
|     connect(m_worker, SIGNAL(disableExit()),this,SLOT(onDisableExit())); | ||||
|     connect(m_worker, SIGNAL(enableExit()),this,SLOT(onEnableExit())); | ||||
|     connect(m_worker, SIGNAL(stopStartTimer()),this,SLOT(onStopStartTimer())); | ||||
|     connect(m_worker, SIGNAL(restartExitTimer()),this,SLOT(onRestartExitTimer())); | ||||
|     connect(m_worker, SIGNAL(appendText(QString,QString)),this,SLOT(onAppendText(QString,QString))); | ||||
|     connect(m_worker, SIGNAL(showErrorMessage(QString,QString)),this, SLOT(onShowErrorMessage(QString,QString))); | ||||
|     connect(m_worker, SIGNAL(showStatusMessage(QString,QString)),this, SLOT(onShowStatusMessage(QString,QString))); | ||||
|     connect(m_worker, SIGNAL(showErrorMessage(QStringList)),this, SLOT(onShowErrorMessage(QStringList))); | ||||
|     connect(m_worker, SIGNAL(showStatusMessage(QString,QString)),this, SLOT(onShowStatusMessage(QString,QString))); | ||||
|     connect(m_worker, SIGNAL(replaceLast(QString,QString)),this,SLOT(onReplaceLast(QString,QString))); | ||||
|     connect(m_worker, SIGNAL(replaceLast(QStringList,QString)),this, SLOT(onReplaceLast(QStringList,QString))); | ||||
| #endif | ||||
| } | ||||
|  | ||||
| MainWindow::~MainWindow() { | ||||
|     //delete m_startTimer; | ||||
|     //delete m_exitTimer; | ||||
|     closeSerialPort(); | ||||
|     delete ui; | ||||
| } | ||||
|  | ||||
| bool MainWindow::openSerialPort() { | ||||
|    // const SettingsDialog::Settings p = m_settings->settings(); | ||||
|    m_serial->setPortName(""); | ||||
|    m_serial->setBaudRate(QSerialPort::Baud115200); | ||||
|    m_serial->setDataBits(QSerialPort::DataBits::Data8); | ||||
|    m_serial->setParity(QSerialPort::Parity::NoParity); | ||||
|    m_serial->setStopBits(QSerialPort::StopBits::OneStop); | ||||
|    m_serial->setFlowControl(QSerialPort::FlowControl::NoFlowControl); | ||||
|    if (m_serial->open(QIODevice::ReadWrite)) { | ||||
|        //m_console->setEnabled(true); | ||||
|        //m_console->setLocalEchoEnabled(p.localEchoEnabled); | ||||
|        //m_ui->actionConnect->setEnabled(false); | ||||
|        //m_ui->actionDisconnect->setEnabled(true); | ||||
|        //m_ui->actionConfigure->setEnabled(false); | ||||
|        //showStatusMessage(tr("Connected to %1 : %2, %3, %4, %5, %6") | ||||
|        //                  .arg(p.name, p.stringBaudRate, p.stringDataBits, | ||||
|        //                       p.stringParity, p.stringStopBits, p.stringFlowControl)); | ||||
|        return true; | ||||
|    } else { | ||||
|        //QMessageBox::critical(this, tr("Error"), m_serial->errorString()); | ||||
|  | ||||
|        //showStatusMessage(tr("Open error")); | ||||
|    } | ||||
|    return false; | ||||
| } | ||||
|  | ||||
| bool MainWindow::closeSerialPort() { | ||||
|     if (m_serial->isOpen()) { | ||||
|         m_serial->close(); | ||||
|         return true; | ||||
|     } | ||||
|     return false; | ||||
|     //m_console->setEnabled(false); | ||||
|     //m_ui->actionConnect->setEnabled(true); | ||||
|     //m_ui->actionDisconnect->setEnabled(false); | ||||
|     //m_ui->actionConfigure->setEnabled(true); | ||||
|     //showStatusMessage(tr("Disconnected")); | ||||
| } | ||||
|  | ||||
| #if 0 | ||||
| void MainWindow::customEvent(QEvent *event) { | ||||
|     if (event->type() == ProgressEvent::type()) { | ||||
|         ProgressEvent *pevent = (ProgressEvent *)event; | ||||
|         int const progress = pevent->progressPercent(); | ||||
|         QObject const *sender = pevent->sender(); | ||||
|         if (sender == this) { | ||||
|             switch(progress) { | ||||
|             case 0: { | ||||
|                 ui->updateProgress->reset(); | ||||
|             } break; | ||||
|             case START_PROGRESS_LOOP: { | ||||
|                 m_progressRunning = true; | ||||
|                 ui->updateProgress->reset(); | ||||
|                 // m_progressValue = 10; | ||||
|                 QApplication::postEvent(this, new ProgressEvent(this, 1)); | ||||
|             } break; | ||||
|             case STOP_PROGRESS_LOOP: { | ||||
|                 m_progressRunning = false; | ||||
|                 // m_progressValue -= 10; | ||||
|                 // m_worker->setProgress(m_progressValue/10); | ||||
|             } break; | ||||
|             default: { | ||||
|                 if (m_progressRunning) { | ||||
|                     // m_progressValue = progress; | ||||
|                     ui->updateProgress->setValue(progress); | ||||
|                     // ueberpruefen: hauptfenster schickt sich selber ein event | ||||
|                     // QApplication::postEvent(this, new ProgressEvent(this, progress)); | ||||
|                     // QThread::msleep(500); | ||||
|                 }} | ||||
|             } | ||||
|         } else | ||||
|         if (sender == m_worker) { | ||||
|             switch(progress) { | ||||
|             case 0: { | ||||
|                 ui->updateProgress->reset(); | ||||
|             } break; | ||||
|             case START_PROGRESS_LOOP: { | ||||
|                 QApplication::postEvent(this, new ProgressEvent(this, START_PROGRESS_LOOP)); | ||||
|             } break; | ||||
|             case STOP_PROGRESS_LOOP: { | ||||
|                 QApplication::postEvent(this, new ProgressEvent(this, STOP_PROGRESS_LOOP)); | ||||
|             } break; | ||||
|             default:{ | ||||
|                 ui->updateProgress->setValue(progress); | ||||
|             }} | ||||
|         } else { | ||||
|             qCritical() << "!!! UNKNOWN SENDER !!!"; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     QThread::yieldCurrentThread(); | ||||
| } | ||||
|  | ||||
| void MainWindow::onStopStartTimer() { | ||||
|     m_startTimer->stop(); | ||||
| } | ||||
|  | ||||
| void MainWindow::onDisableExit() { | ||||
|    ui->exit->setEnabled(false); | ||||
| } | ||||
|  | ||||
| void MainWindow::onEnableExit() { | ||||
|    ui->exit->setEnabled(true); | ||||
| } | ||||
|  | ||||
| void MainWindow::onRestartExitTimer() { | ||||
|     m_exitTimer->stop(); | ||||
|     m_exitTimer->start(5 * 1000); | ||||
|  | ||||
|     scrollDownTextEdit(); | ||||
|     ui->updateStatus->setEnabled(false); | ||||
| } | ||||
|  | ||||
| void MainWindow::onQuit() { | ||||
|     m_exitTimer->stop(); | ||||
|     int errorCode = 0; | ||||
|  | ||||
|     qCritical() | ||||
|         << QString("ON QUIT: CURRENT STEP %1") | ||||
|             .arg(m_worker->getSmap()[m_worker->currentStep()]); | ||||
|  | ||||
|     // TODO: replace SEND_LAST_VERSION with UPDATE_SUCCEEDED | ||||
|     if (m_worker->currentStep() != Worker::UPDATE_STEP::SEND_LAST_VERSION) { | ||||
|         errorCode = -1; | ||||
|     } | ||||
|     qCritical() << QString("ON QUIT: EXIT CODE %1").arg(errorCode); | ||||
|     qApp->exit(errorCode); | ||||
| } | ||||
|  | ||||
| void MainWindow::scrollDownTextEdit() { | ||||
|     // Utils::printInfoMsg(QString("SCROLL-DOWN-TEXT_EDIT CALLED AT ") | ||||
|     //    + QDateTime::currentDateTime().toString(Qt::ISODateWithMs)); | ||||
|  | ||||
|     ui->updateStatus->setEnabled(true); | ||||
|  | ||||
|     QTextCursor tmpCursor = ui->updateStatus->textCursor(); | ||||
|     tmpCursor.movePosition(QTextCursor::End); | ||||
|     ui->updateStatus->setTextCursor(tmpCursor); | ||||
|     ui->updateStatus->ensureCursorVisible(); | ||||
| } | ||||
|  | ||||
| void MainWindow::onAppendText(QString text, QString suffix) { | ||||
|     // Utils::printInfoMsg(QString("ON APPEND CALLED AT ") | ||||
|     //    + QDateTime::currentDateTime().toString(Qt::ISODateWithMs)); | ||||
|  | ||||
|     QString editText = ui->updateStatus->toPlainText(); | ||||
|     scrollDownTextEdit(); | ||||
|  | ||||
|     if (!suffix.isNull() && suffix.size() > 0) { | ||||
|         //qInfo() << "TEXT" << text << "SUFFIX" << suffix; | ||||
|         if (suffix == Worker::UPDATE_STEP_SUCCESS || suffix == Worker::UPDATE_STEP_FAIL) { | ||||
|             ui->updateStatus->insertPlainText(QString("\n").leftJustified(m_width-3, '=') + " "); | ||||
|             // editText += QString("\n").leftJustified(m_width-3, '='); | ||||
|             // editText += " "; | ||||
|         } | ||||
|         QString const &add = (QString("\n") + text).leftJustified(m_width - (2 + suffix.size())) + suffix; | ||||
|         ui->updateStatus->insertPlainText(add); | ||||
|         // editText += add; | ||||
|     } else { | ||||
|         QString const &add = text.leftJustified(m_width-9); | ||||
|         ui->updateStatus->insertPlainText(add); | ||||
|         //editText += add; | ||||
|     } | ||||
|  | ||||
|     // debug | ||||
|     // QString editText = ui->updateStatus->toPlainText(); | ||||
|     // Utils::printLineEditInfo(editText.split('\n', QString::SplitBehavior::SkipEmptyParts)); | ||||
|     // ui->updateStatus->setText(editText.trimmed()); | ||||
|  | ||||
|     scrollDownTextEdit(); | ||||
| } | ||||
|  | ||||
| void MainWindow::onReplaceLast(QStringList newTextLines, QString suffix) { | ||||
|     // Utils::printInfoMsg(QString("ON REPLACE LAST (LIST) CALLED AT ") | ||||
|     //    + QDateTime::currentDateTime().toString(Qt::ISODateWithMs)); | ||||
|     int const s = newTextLines.size(); | ||||
|     if (s > 0) { | ||||
|         QString editText = ui->updateStatus->toPlainText(); | ||||
|         QStringList lines = editText.split('\n', QString::SplitBehavior::SkipEmptyParts); | ||||
|         QString newText; | ||||
|         if (lines.size() >= s) { | ||||
|             for (int i = 0; i < s; ++i) { | ||||
|                 lines.removeLast(); | ||||
|             } | ||||
|             if (lines.size() > 0) { | ||||
|                 newText = lines.join('\n'); | ||||
|                 newText += '\n'; | ||||
|             } | ||||
|             QStringList newLines; | ||||
|             for (int i = 0; i < s; ++i) { | ||||
|                 if (i == 0 && !suffix.isNull() && suffix.size() > 0 && suffix != "\n") { | ||||
|                     newLines += Utils::rstrip(newTextLines.at(i).leftJustified(m_width-10) + suffix); | ||||
|                 } else { | ||||
|                     newLines += Utils::rstrip(newTextLines.at(i).leftJustified(m_width-10)); | ||||
|                 } | ||||
|             } | ||||
|             lines += newLines; | ||||
|             newText += newLines.join(' '); | ||||
|         } | ||||
|  | ||||
|         ui->updateStatus->setText(newText); | ||||
|         Utils::printLineEditInfo(lines); | ||||
|         scrollDownTextEdit(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void MainWindow::onReplaceLast(QString text, QString suffix) { | ||||
|     // Utils::printInfoMsg(QString("ON REPLACE LAST (TEXT) CALLED AT ") | ||||
|     //    + QDateTime::currentDateTime().toString(Qt::ISODateWithMs)); | ||||
|     QString editText = ui->updateStatus->toPlainText(); | ||||
|     QStringList lines = editText.split('\n', QString::SplitBehavior::SkipEmptyParts); | ||||
|     if (lines.size() > 0) { | ||||
|         // removing the last line is really meant for refreshing the last line | ||||
|         // with a string very similar than the original one, typically only | ||||
|         // followed by a suffix. | ||||
|         if (lines.last().contains(text)) { | ||||
|             lines.removeLast(); | ||||
|         } | ||||
|         if (!suffix.isNull() && suffix.size() > 0 && suffix != "\n") { | ||||
|             QString const add = text.leftJustified(m_width-10) + suffix; | ||||
|             if (!add.isEmpty()) { | ||||
|                 lines += text.leftJustified(m_width-10) + suffix; | ||||
|             } | ||||
|         } else { | ||||
|             QString const add = text.leftJustified(m_width-10); | ||||
|             if (!add.isEmpty()) { | ||||
|                 lines += text.leftJustified(m_width-10); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     Utils::printLineEditInfo(lines); | ||||
|     ui->updateStatus->setText(lines.join('\n').trimmed()); | ||||
|     scrollDownTextEdit(); | ||||
| } | ||||
|  | ||||
| void MainWindow::onShowMessage(QString title, QString text) { | ||||
|     this->statusBar()->clearMessage(); | ||||
|     this->statusBar()->showMessage( // timeout: 10000 | ||||
|         QString(title + " " + text).leftJustified(80, ' '), 10000); | ||||
| } | ||||
|  | ||||
| void MainWindow::onShowErrorMessage(QString title, QString text) { | ||||
|     onShowMessage(title, text); | ||||
| } | ||||
|  | ||||
| void MainWindow::onShowStatusMessage(QString title, QString text) { | ||||
|     onShowMessage(title, text); | ||||
| } | ||||
|  | ||||
| void MainWindow::onShowErrorMessage(QStringList lst) { | ||||
|     if (lst.size() >= 2) { | ||||
|         onShowMessage(lst.at(0), lst.at(1)); | ||||
|     } | ||||
|     if (lst.size() == 1) { | ||||
|         onShowMessage(lst.at(0), ""); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void MainWindow::onShowStatusMessage(QStringList lst) { | ||||
|     if (lst.size() >= 2) { | ||||
|         onShowMessage(lst.at(0), lst.at(1)); | ||||
|     } | ||||
|     if (lst.size() == 1) { | ||||
|         onShowMessage(lst.at(0), ""); | ||||
|     } | ||||
| } | ||||
| #endif | ||||
							
								
								
									
										58
									
								
								DownloadDCFirmware/mainwindow.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								DownloadDCFirmware/mainwindow.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | ||||
| #ifndef MAINWINDOW_H | ||||
| #define MAINWINDOW_H | ||||
|  | ||||
| #include <QMainWindow> | ||||
| #include <QTimer> | ||||
| #include <QStatusBar> | ||||
| #include <QWidget> | ||||
| #include <QSerialPort> | ||||
| #include <QSerialPortInfo> | ||||
|  | ||||
| QT_BEGIN_NAMESPACE | ||||
| namespace Ui { class MainWindow; } | ||||
| QT_END_NAMESPACE | ||||
|  | ||||
|  | ||||
| class MainWindow : public QMainWindow { | ||||
|     Q_OBJECT | ||||
|  | ||||
| public: | ||||
|     MainWindow(QWidget *parent = nullptr); | ||||
|     ~MainWindow(); | ||||
|  | ||||
| public slots: | ||||
|     //void onAppendText(QString, QString suffix = ""); | ||||
|     //void onReplaceLast(QStringList, QString suffix = ""); | ||||
|     //void onReplaceLast(QString, QString suffix = ""); | ||||
|     //void onShowErrorMessage(QString, QString); | ||||
|     //void onShowStatusMessage(QString, QString); | ||||
|     //void onShowErrorMessage(QStringList); | ||||
|     //void onShowStatusMessage(QStringList); | ||||
|     //void onStopStartTimer(); | ||||
|     //void onRestartExitTimer(); | ||||
|     //void onEnableExit(); | ||||
|     //void onDisableExit(); | ||||
|  | ||||
| signals: | ||||
|  | ||||
| private slots: | ||||
| //    void onQuit(); | ||||
|  | ||||
| private: | ||||
| //    void scrollDownTextEdit(); | ||||
| //    void onShowMessage(QString, QString); | ||||
|  | ||||
|     bool openSerialPort(); | ||||
|     bool closeSerialPort(); | ||||
|  | ||||
|     Ui::MainWindow *ui; | ||||
|     QSerialPort *m_serial; | ||||
| //    Worker *m_worker; | ||||
|     int const m_width; | ||||
| //    QTimer *m_startTimer; | ||||
| //    QTimer *m_exitTimer; | ||||
| //    bool m_progressRunning; | ||||
|     //int m_progressValue; | ||||
| //    UpdateDcEvent::UpdateStep m_updateStep; | ||||
| }; | ||||
| #endif // MAINWINDOW_H | ||||
							
								
								
									
										62
									
								
								DownloadDCFirmware/mainwindow.ui
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								DownloadDCFirmware/mainwindow.ui
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,62 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <ui version="4.0"> | ||||
|  <class>MainWindow</class> | ||||
|  <widget class="QMainWindow" name="MainWindow"> | ||||
|   <property name="geometry"> | ||||
|    <rect> | ||||
|     <x>0</x> | ||||
|     <y>0</y> | ||||
|     <width>800</width> | ||||
|     <height>480</height> | ||||
|    </rect> | ||||
|   </property> | ||||
|   <property name="font"> | ||||
|    <font> | ||||
|     <family>Source Code Pro</family> | ||||
|    </font> | ||||
|   </property> | ||||
|   <property name="windowTitle"> | ||||
|    <string>MainWindow</string> | ||||
|   </property> | ||||
|   <widget class="QWidget" name="centralwidget"> | ||||
|    <widget class="QWidget" name="layoutWidget"> | ||||
|     <property name="geometry"> | ||||
|      <rect> | ||||
|       <x>10</x> | ||||
|       <y>10</y> | ||||
|       <width>781</width> | ||||
|       <height>441</height> | ||||
|      </rect> | ||||
|     </property> | ||||
|     <layout class="QGridLayout" name="gridLayout"> | ||||
|      <item row="0" column="0" rowspan="3" colspan="2"> | ||||
|       <widget class="QTextEdit" name="updateStatus"> | ||||
|        <property name="enabled"> | ||||
|         <bool>true</bool> | ||||
|        </property> | ||||
|        <property name="font"> | ||||
|         <font> | ||||
|          <family>Misc Fixed</family> | ||||
|          <pointsize>11</pointsize> | ||||
|          <weight>75</weight> | ||||
|          <bold>true</bold> | ||||
|         </font> | ||||
|        </property> | ||||
|        <property name="verticalScrollBarPolicy"> | ||||
|         <enum>Qt::ScrollBarAsNeeded</enum> | ||||
|        </property> | ||||
|        <property name="horizontalScrollBarPolicy"> | ||||
|         <enum>Qt::ScrollBarAsNeeded</enum> | ||||
|        </property> | ||||
|        <property name="sizeAdjustPolicy"> | ||||
|         <enum>QAbstractScrollArea::AdjustToContents</enum> | ||||
|        </property> | ||||
|       </widget> | ||||
|      </item> | ||||
|     </layout> | ||||
|    </widget> | ||||
|   </widget> | ||||
|  </widget> | ||||
|  <resources/> | ||||
|  <connections/> | ||||
| </ui> | ||||
							
								
								
									
										0
									
								
								DownloadDCFirmware/receiver_thread.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								DownloadDCFirmware/receiver_thread.cpp
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										0
									
								
								DownloadDCFirmware/receiver_thread.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								DownloadDCFirmware/receiver_thread.h
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										0
									
								
								DownloadDCFirmware/sender_thread.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								DownloadDCFirmware/sender_thread.cpp
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										0
									
								
								DownloadDCFirmware/sender_thread.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								DownloadDCFirmware/sender_thread.h
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										13
									
								
								DownloadDCFirmware/worker_thread.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								DownloadDCFirmware/worker_thread.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| #include "worker_thread.h" | ||||
|  | ||||
| WorkerThread::WorkerThread(QObject *parent) | ||||
|   : QThread(parent) { | ||||
| } | ||||
|  | ||||
| WorkerThread::~WorkerThread() { | ||||
|  | ||||
| } | ||||
|  | ||||
| void WorkerThread::run() { | ||||
|  | ||||
| } | ||||
							
								
								
									
										20
									
								
								DownloadDCFirmware/worker_thread.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								DownloadDCFirmware/worker_thread.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| #ifndef WORKER_THREAD_H_INCLUDED | ||||
| #define WORKER_THREAD_H_INCLUDED | ||||
|  | ||||
| #include <QThread> | ||||
|  | ||||
| class WorkerThread : public QThread { | ||||
|     Q_OBJECT | ||||
|  | ||||
| public: | ||||
|     explicit WorkerThread(QObject *parent = nullptr); | ||||
|     ~WorkerThread(); | ||||
|  | ||||
|  | ||||
| private: | ||||
|     void run() override; | ||||
|  | ||||
|     bool m_quit = false; | ||||
| }; | ||||
|  | ||||
| #endif // WORKER_THREAD_H_INCLUDED | ||||
							
								
								
									
										0
									
								
								DownloadDCJsonFiles/ATBDownloadDCJsonFiles.ini
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								DownloadDCJsonFiles/ATBDownloadDCJsonFiles.ini
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										102
									
								
								DownloadDCJsonFiles/DownloadDCJsonFiles.pro
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										102
									
								
								DownloadDCJsonFiles/DownloadDCJsonFiles.pro
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,102 @@ | ||||
| QT += core | ||||
| QT += serialport network | ||||
|  | ||||
| TARGET = ATBDownloadDCJsonFiles | ||||
|  | ||||
| VERSION="0.1.0" | ||||
| win32 { | ||||
|     BUILD_DATE=$$system("date /t") | ||||
|     BUILD_TIME=$$system("time /t") | ||||
| } else { | ||||
|     BUILD_DATE=$$system("date +%d-%m-%y") | ||||
|     BUILD_TIME=$$system("date +%H:%M:%S") | ||||
| } | ||||
|  | ||||
| GIT_COMMIT=$$system("git log -1 --format=oneline | cut -d' ' -f1") | ||||
|  | ||||
| EXTENDED_VERSION="$${VERSION}-$${GIT_COMMIT}" | ||||
|  | ||||
| !contains(CONFIG, INCLUDEINTERFACES) { | ||||
|     INCLUDEINTERFACES=/opt/ptu5/opt/DCLibraries/include | ||||
| } | ||||
|  | ||||
| INCLUDEPATH += \ | ||||
|     plugins \ | ||||
|     $${INCLUDEINTERFACES} \ | ||||
|     $${_PRO_FILE_PWD_}/../UpdatePTUDevCtrl \ | ||||
|     $${_PRO_FILE_PWD_}/../../ATBUpdateTool/common/include | ||||
|  | ||||
| CONFIG += c++17 | ||||
|  | ||||
| DEFINES+=APP_VERSION=\\\"$$VERSION\\\" | ||||
| DEFINES+=APP_BUILD_DATE=\\\"$$BUILD_DATE\\\" | ||||
| DEFINES+=APP_BUILD_TIME=\\\"$$BUILD_TIME\\\" | ||||
| DEFINES+=APP_EXTENDED_VERSION=\\\"$$EXTENDED_VERSION\\\" | ||||
|  | ||||
| # keep comments, as /* fall through */ | ||||
| QMAKE_CXXFLAGS += -C | ||||
| QMAKE_CXXFLAGS += -g | ||||
| QMAKE_CXXFLAGS += -Wno-deprecated-copy -O | ||||
|  | ||||
| contains( CONFIG, PTU5 ) { | ||||
|     greaterThan(QT_MAJOR_VERSION, 4): QT += serialport | ||||
|     CONFIG += link_pkgconfig | ||||
|     lessThan(QT_MAJOR_VERSION, 5):   PKGCONFIG += qextserialport | ||||
|     QMAKE_CXXFLAGS += -O2 -std=c++17   # for GCC >= 4.7 | ||||
|     # QMAKE_CXXFLAGS += -Wno-deprecated-copy | ||||
|  | ||||
|     PTU5BASEPATH = /opt/devel/ptu5 | ||||
|  | ||||
|     INCLUDEPATH += $$PTU5BASEPATH/qt/libs/devicecontroller/include | ||||
|     LIBS += -L$$PTU5BASEPATH/qt/libs/devicecontroller/library | ||||
|  | ||||
|     ARCH = PTU5 | ||||
|     DEFINES+=PTU5 | ||||
| } | ||||
| contains( CONFIG, PTU5_YOCTO ) { | ||||
|     greaterThan(QT_MAJOR_VERSION, 4): QT += serialport | ||||
|     QMAKE_CXXFLAGS += -std=c++17   # for GCC >= 4.7 | ||||
|     # QMAKE_CXXFLAGS += -Wno-deprecated-copy | ||||
|     PTU5BASEPATH = /opt/devel/ptu5 | ||||
|     ARCH = PTU5 | ||||
|     DEFINES+=PTU5 | ||||
|  | ||||
|     # add qmqtt lib | ||||
|     #LIBS += -lQt5Qmqtt | ||||
| } | ||||
| contains( CONFIG, DesktopLinux ) { | ||||
|     greaterThan(QT_MAJOR_VERSION, 4): QT += serialport | ||||
|     lessThan(QT_MAJOR_VERSION, 5):    CONFIG += extserialport | ||||
|     # QMAKE_CC = ccache $$QMAKE_CC | ||||
|     # QMAKE_CXX = ccache $$QMAKE_CXX | ||||
|     QMAKE_CXXFLAGS += -std=c++17 | ||||
|     # QMAKE_CXXFLAGS += -Wno-deprecated-copy | ||||
|     linux-clang {  QMAKE_CXXFLAGS += -Qunused-arguments   } | ||||
|     ARCH = DesktopLinux | ||||
|     DEFINES+=DesktopLinux | ||||
| } | ||||
|  | ||||
| SOURCES += \ | ||||
|         main.cpp \ | ||||
|         ../UpdatePTUDevCtrl/message_handler.cpp \ | ||||
|         ../UpdatePTUDevCtrl/commandline_parser.cpp \ | ||||
|         update.cpp \ | ||||
|         ../../ATBUpdateTool/common/src/System.cpp | ||||
|  | ||||
| HEADERS += \ | ||||
|         ../UpdatePTUDevCtrl/message_handler.h \ | ||||
|         ../UpdatePTUDevCtrl/commandline_parser.h \ | ||||
|         update.h \ | ||||
|         ../../ATBUpdateTool/common/include/System.h | ||||
|  | ||||
| OTHER_FILES += \ | ||||
|     ATBDownloadDCJsonFiles.ini | ||||
|  | ||||
|  | ||||
| ########################################################################################## | ||||
| # for running program on target through QtCreator | ||||
| contains( CONFIG, PTU5 ) { | ||||
|    qnx: target.path = /tmp/$${TARGET}/bin | ||||
|    else: unix:!android: target.path = /opt/app/tools/atbupdate/ | ||||
|    !isEmpty(target.path): INSTALLS += target | ||||
| } | ||||
							
								
								
									
										445
									
								
								DownloadDCJsonFiles/DownloadDCJsonFiles.pro.user
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										445
									
								
								DownloadDCJsonFiles/DownloadDCJsonFiles.pro.user
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,445 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <!DOCTYPE QtCreatorProject> | ||||
| <!-- Written by QtCreator 5.0.2, 2024-02-28T16:01:22. --> | ||||
| <qtcreator> | ||||
|  <data> | ||||
|   <variable>EnvironmentId</variable> | ||||
|   <value type="QByteArray">{6a494cc5-6dea-4681-86fc-d47b9761a1f4}</value> | ||||
|  </data> | ||||
|  <data> | ||||
|   <variable>ProjectExplorer.Project.ActiveTarget</variable> | ||||
|   <value type="int">0</value> | ||||
|  </data> | ||||
|  <data> | ||||
|   <variable>ProjectExplorer.Project.EditorSettings</variable> | ||||
|   <valuemap type="QVariantMap"> | ||||
|    <value type="bool" key="EditorConfiguration.AutoIndent">true</value> | ||||
|    <value type="bool" key="EditorConfiguration.AutoSpacesForTabs">false</value> | ||||
|    <value type="bool" key="EditorConfiguration.CamelCaseNavigation">true</value> | ||||
|    <valuemap type="QVariantMap" key="EditorConfiguration.CodeStyle.0"> | ||||
|     <value type="QString" key="language">Cpp</value> | ||||
|     <valuemap type="QVariantMap" key="value"> | ||||
|      <value type="QByteArray" key="CurrentPreferences">CppGlobal</value> | ||||
|     </valuemap> | ||||
|    </valuemap> | ||||
|    <valuemap type="QVariantMap" key="EditorConfiguration.CodeStyle.1"> | ||||
|     <value type="QString" key="language">QmlJS</value> | ||||
|     <valuemap type="QVariantMap" key="value"> | ||||
|      <value type="QByteArray" key="CurrentPreferences">QmlJSGlobal</value> | ||||
|     </valuemap> | ||||
|    </valuemap> | ||||
|    <value type="int" key="EditorConfiguration.CodeStyle.Count">2</value> | ||||
|    <value type="QByteArray" key="EditorConfiguration.Codec">UTF-8</value> | ||||
|    <value type="bool" key="EditorConfiguration.ConstrainTooltips">false</value> | ||||
|    <value type="int" key="EditorConfiguration.IndentSize">4</value> | ||||
|    <value type="bool" key="EditorConfiguration.KeyboardTooltips">false</value> | ||||
|    <value type="int" key="EditorConfiguration.MarginColumn">80</value> | ||||
|    <value type="bool" key="EditorConfiguration.MouseHiding">true</value> | ||||
|    <value type="bool" key="EditorConfiguration.MouseNavigation">true</value> | ||||
|    <value type="int" key="EditorConfiguration.PaddingMode">1</value> | ||||
|    <value type="bool" key="EditorConfiguration.PreferSingleLineComments">false</value> | ||||
|    <value type="bool" key="EditorConfiguration.ScrollWheelZooming">true</value> | ||||
|    <value type="bool" key="EditorConfiguration.ShowMargin">false</value> | ||||
|    <value type="int" key="EditorConfiguration.SmartBackspaceBehavior">0</value> | ||||
|    <value type="bool" key="EditorConfiguration.SmartSelectionChanging">true</value> | ||||
|    <value type="bool" key="EditorConfiguration.SpacesForTabs">true</value> | ||||
|    <value type="int" key="EditorConfiguration.TabKeyBehavior">0</value> | ||||
|    <value type="int" key="EditorConfiguration.TabSize">8</value> | ||||
|    <value type="bool" key="EditorConfiguration.UseGlobal">true</value> | ||||
|    <value type="bool" key="EditorConfiguration.UseIndenter">false</value> | ||||
|    <value type="int" key="EditorConfiguration.Utf8BomBehavior">1</value> | ||||
|    <value type="bool" key="EditorConfiguration.addFinalNewLine">true</value> | ||||
|    <value type="bool" key="EditorConfiguration.cleanIndentation">true</value> | ||||
|    <value type="bool" key="EditorConfiguration.cleanWhitespace">true</value> | ||||
|    <value type="QString" key="EditorConfiguration.ignoreFileTypes">*.md, *.MD, Makefile</value> | ||||
|    <value type="bool" key="EditorConfiguration.inEntireDocument">false</value> | ||||
|    <value type="bool" key="EditorConfiguration.skipTrailingWhitespace">true</value> | ||||
|   </valuemap> | ||||
|  </data> | ||||
|  <data> | ||||
|   <variable>ProjectExplorer.Project.PluginSettings</variable> | ||||
|   <valuemap type="QVariantMap"> | ||||
|    <valuemap type="QVariantMap" key="AutoTest.ActiveFrameworks"> | ||||
|     <value type="bool" key="AutoTest.Framework.Boost">true</value> | ||||
|     <value type="bool" key="AutoTest.Framework.CTest">false</value> | ||||
|     <value type="bool" key="AutoTest.Framework.Catch">true</value> | ||||
|     <value type="bool" key="AutoTest.Framework.GTest">true</value> | ||||
|     <value type="bool" key="AutoTest.Framework.QtQuickTest">true</value> | ||||
|     <value type="bool" key="AutoTest.Framework.QtTest">true</value> | ||||
|    </valuemap> | ||||
|    <valuemap type="QVariantMap" key="AutoTest.CheckStates"/> | ||||
|    <value type="int" key="AutoTest.RunAfterBuild">0</value> | ||||
|    <value type="bool" key="AutoTest.UseGlobal">true</value> | ||||
|    <valuemap type="QVariantMap" key="ClangTools"> | ||||
|     <value type="bool" key="ClangTools.AnalyzeOpenFiles">true</value> | ||||
|     <value type="bool" key="ClangTools.BuildBeforeAnalysis">true</value> | ||||
|     <value type="QString" key="ClangTools.DiagnosticConfig">Builtin.DefaultTidyAndClazy</value> | ||||
|     <value type="int" key="ClangTools.ParallelJobs">3</value> | ||||
|     <valuelist type="QVariantList" key="ClangTools.SelectedDirs"/> | ||||
|     <valuelist type="QVariantList" key="ClangTools.SelectedFiles"/> | ||||
|     <valuelist type="QVariantList" key="ClangTools.SuppressedDiagnostics"/> | ||||
|     <value type="bool" key="ClangTools.UseGlobalSettings">true</value> | ||||
|    </valuemap> | ||||
|   </valuemap> | ||||
|  </data> | ||||
|  <data> | ||||
|   <variable>ProjectExplorer.Project.Target.0</variable> | ||||
|   <valuemap type="QVariantMap"> | ||||
|    <value type="QString" key="DeviceType">GenericLinuxOsType</value> | ||||
|    <value type="QString" key="ProjectExplorer.ProjectConfiguration.DefaultDisplayName">Yocto i.MX6-ATB-PTU5</value> | ||||
|    <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Yocto i.MX6-ATB-PTU5</value> | ||||
|    <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">{804f60e1-6e88-41af-b072-9f5c6a606099}</value> | ||||
|    <value type="int" key="ProjectExplorer.Target.ActiveBuildConfiguration">0</value> | ||||
|    <value type="int" key="ProjectExplorer.Target.ActiveDeployConfiguration">0</value> | ||||
|    <value type="int" key="ProjectExplorer.Target.ActiveRunConfiguration">0</value> | ||||
|    <valuemap type="QVariantMap" key="ProjectExplorer.Target.BuildConfiguration.0"> | ||||
|     <value type="int" key="EnableQmlDebugging">0</value> | ||||
|     <value type="QString" key="ProjectExplorer.BuildConfiguration.BuildDirectory">/opt/ptu5/opt/build-DownloadDCJsonFiles-Yocto_i_MX6_ATB_PTU5-Debug</value> | ||||
|     <value type="QString" key="ProjectExplorer.BuildConfiguration.BuildDirectory.shadowDir">/opt/ptu5/opt/build-DownloadDCJsonFiles-Yocto_i_MX6_ATB_PTU5-Debug</value> | ||||
|     <valuemap type="QVariantMap" key="ProjectExplorer.BuildConfiguration.BuildStepList.0"> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.0"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">QtProjectManager.QMakeBuildStep</value> | ||||
|       <value type="QString" key="QtProjectManager.QMakeBuildStep.QMakeArguments">CONFIG+=PTU5</value> | ||||
|       <value type="bool" key="QtProjectManager.QMakeBuildStep.QMakeForced">false</value> | ||||
|       <valuelist type="QVariantList" key="QtProjectManager.QMakeBuildStep.SelectedAbis"/> | ||||
|      </valuemap> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.1"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">Qt4ProjectManager.MakeStep</value> | ||||
|       <value type="QString" key="Qt4ProjectManager.MakeStep.MakeCommand">/usr/bin/make</value> | ||||
|      </valuemap> | ||||
|      <value type="int" key="ProjectExplorer.BuildStepList.StepsCount">2</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DefaultDisplayName">Build</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Build</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">ProjectExplorer.BuildSteps.Build</value> | ||||
|     </valuemap> | ||||
|     <valuemap type="QVariantMap" key="ProjectExplorer.BuildConfiguration.BuildStepList.1"> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.0"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">Qt4ProjectManager.MakeStep</value> | ||||
|       <value type="QString" key="Qt4ProjectManager.MakeStep.MakeArguments">clean</value> | ||||
|       <value type="QString" key="Qt4ProjectManager.MakeStep.MakeCommand">/usr/bin/make</value> | ||||
|      </valuemap> | ||||
|      <value type="int" key="ProjectExplorer.BuildStepList.StepsCount">1</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DefaultDisplayName">Clean</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Clean</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">ProjectExplorer.BuildSteps.Clean</value> | ||||
|     </valuemap> | ||||
|     <value type="int" key="ProjectExplorer.BuildConfiguration.BuildStepListCount">2</value> | ||||
|     <value type="bool" key="ProjectExplorer.BuildConfiguration.ClearSystemEnvironment">false</value> | ||||
|     <valuelist type="QVariantList" key="ProjectExplorer.BuildConfiguration.CustomParsers"/> | ||||
|     <valuelist type="QVariantList" key="ProjectExplorer.BuildConfiguration.UserEnvironmentChanges"/> | ||||
|     <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Debug</value> | ||||
|     <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">Qt4ProjectManager.Qt4BuildConfiguration</value> | ||||
|     <value type="int" key="Qt4ProjectManager.Qt4BuildConfiguration.BuildConfiguration">2</value> | ||||
|    </valuemap> | ||||
|    <valuemap type="QVariantMap" key="ProjectExplorer.Target.BuildConfiguration.1"> | ||||
|     <value type="QString" key="ProjectExplorer.BuildConfiguration.BuildDirectory">/opt/ptu5/opt/build-DownloadDCJsonFiles-Yocto_i_MX6_ATB_PTU5-Release</value> | ||||
|     <value type="QString" key="ProjectExplorer.BuildConfiguration.BuildDirectory.shadowDir">/opt/ptu5/opt/build-DownloadDCJsonFiles-Yocto_i_MX6_ATB_PTU5-Release</value> | ||||
|     <valuemap type="QVariantMap" key="ProjectExplorer.BuildConfiguration.BuildStepList.0"> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.0"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">QtProjectManager.QMakeBuildStep</value> | ||||
|       <value type="bool" key="QtProjectManager.QMakeBuildStep.QMakeForced">false</value> | ||||
|       <valuelist type="QVariantList" key="QtProjectManager.QMakeBuildStep.SelectedAbis"/> | ||||
|      </valuemap> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.1"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">Qt4ProjectManager.MakeStep</value> | ||||
|      </valuemap> | ||||
|      <value type="int" key="ProjectExplorer.BuildStepList.StepsCount">2</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DefaultDisplayName">Build</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Build</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">ProjectExplorer.BuildSteps.Build</value> | ||||
|     </valuemap> | ||||
|     <valuemap type="QVariantMap" key="ProjectExplorer.BuildConfiguration.BuildStepList.1"> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.0"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">Qt4ProjectManager.MakeStep</value> | ||||
|       <value type="QString" key="Qt4ProjectManager.MakeStep.MakeArguments">clean</value> | ||||
|      </valuemap> | ||||
|      <value type="int" key="ProjectExplorer.BuildStepList.StepsCount">1</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DefaultDisplayName">Clean</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Clean</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">ProjectExplorer.BuildSteps.Clean</value> | ||||
|     </valuemap> | ||||
|     <value type="int" key="ProjectExplorer.BuildConfiguration.BuildStepListCount">2</value> | ||||
|     <value type="bool" key="ProjectExplorer.BuildConfiguration.ClearSystemEnvironment">false</value> | ||||
|     <valuelist type="QVariantList" key="ProjectExplorer.BuildConfiguration.CustomParsers"/> | ||||
|     <valuelist type="QVariantList" key="ProjectExplorer.BuildConfiguration.UserEnvironmentChanges"/> | ||||
|     <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Release</value> | ||||
|     <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">Qt4ProjectManager.Qt4BuildConfiguration</value> | ||||
|     <value type="int" key="Qt4ProjectManager.Qt4BuildConfiguration.BuildConfiguration">0</value> | ||||
|     <value type="int" key="QtQuickCompiler">0</value> | ||||
|    </valuemap> | ||||
|    <valuemap type="QVariantMap" key="ProjectExplorer.Target.BuildConfiguration.2"> | ||||
|     <value type="int" key="EnableQmlDebugging">0</value> | ||||
|     <value type="QString" key="ProjectExplorer.BuildConfiguration.BuildDirectory">/opt/ptu5/opt/build-DownloadDCJsonFiles-Yocto_i_MX6_ATB_PTU5-Profile</value> | ||||
|     <value type="QString" key="ProjectExplorer.BuildConfiguration.BuildDirectory.shadowDir">/opt/ptu5/opt/build-DownloadDCJsonFiles-Yocto_i_MX6_ATB_PTU5-Profile</value> | ||||
|     <valuemap type="QVariantMap" key="ProjectExplorer.BuildConfiguration.BuildStepList.0"> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.0"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">QtProjectManager.QMakeBuildStep</value> | ||||
|       <value type="bool" key="QtProjectManager.QMakeBuildStep.QMakeForced">false</value> | ||||
|       <valuelist type="QVariantList" key="QtProjectManager.QMakeBuildStep.SelectedAbis"/> | ||||
|      </valuemap> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.1"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">Qt4ProjectManager.MakeStep</value> | ||||
|      </valuemap> | ||||
|      <value type="int" key="ProjectExplorer.BuildStepList.StepsCount">2</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DefaultDisplayName">Build</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Build</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">ProjectExplorer.BuildSteps.Build</value> | ||||
|     </valuemap> | ||||
|     <valuemap type="QVariantMap" key="ProjectExplorer.BuildConfiguration.BuildStepList.1"> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.0"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">Qt4ProjectManager.MakeStep</value> | ||||
|       <value type="QString" key="Qt4ProjectManager.MakeStep.MakeArguments">clean</value> | ||||
|      </valuemap> | ||||
|      <value type="int" key="ProjectExplorer.BuildStepList.StepsCount">1</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DefaultDisplayName">Clean</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Clean</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">ProjectExplorer.BuildSteps.Clean</value> | ||||
|     </valuemap> | ||||
|     <value type="int" key="ProjectExplorer.BuildConfiguration.BuildStepListCount">2</value> | ||||
|     <value type="bool" key="ProjectExplorer.BuildConfiguration.ClearSystemEnvironment">false</value> | ||||
|     <valuelist type="QVariantList" key="ProjectExplorer.BuildConfiguration.CustomParsers"/> | ||||
|     <valuelist type="QVariantList" key="ProjectExplorer.BuildConfiguration.UserEnvironmentChanges"/> | ||||
|     <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Profile</value> | ||||
|     <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">Qt4ProjectManager.Qt4BuildConfiguration</value> | ||||
|     <value type="int" key="Qt4ProjectManager.Qt4BuildConfiguration.BuildConfiguration">0</value> | ||||
|     <value type="int" key="QtQuickCompiler">0</value> | ||||
|     <value type="int" key="SeparateDebugInfo">0</value> | ||||
|    </valuemap> | ||||
|    <value type="int" key="ProjectExplorer.Target.BuildConfigurationCount">3</value> | ||||
|    <valuemap type="QVariantMap" key="ProjectExplorer.Target.DeployConfiguration.0"> | ||||
|     <valuemap type="QVariantMap" key="ProjectExplorer.BuildConfiguration.BuildStepList.0"> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.0"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">RemoteLinux.CheckForFreeDiskSpaceStep</value> | ||||
|       <valuelist type="QVariantList" key="ProjectExplorer.RunConfiguration.LastDeployedFiles"/> | ||||
|       <valuelist type="QVariantList" key="ProjectExplorer.RunConfiguration.LastDeployedHosts"/> | ||||
|       <valuelist type="QVariantList" key="ProjectExplorer.RunConfiguration.LastDeployedRemotePaths"/> | ||||
|       <valuelist type="QVariantList" key="ProjectExplorer.RunConfiguration.LastDeployedSysroots"/> | ||||
|       <value type="QString" key="RemoteLinux.CheckForFreeDiskSpaceStep.PathToCheck">/</value> | ||||
|       <value type="qlonglong" key="RemoteLinux.CheckForFreeDiskSpaceStep.RequiredSpace">5242880</value> | ||||
|       <valuelist type="QVariantList" key="RemoteLinux.LastDeployedLocalTimes"/> | ||||
|       <valuelist type="QVariantList" key="RemoteLinux.LastDeployedRemoteTimes"/> | ||||
|      </valuemap> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.1"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">RemoteLinux.KillAppStep</value> | ||||
|       <valuelist type="QVariantList" key="ProjectExplorer.RunConfiguration.LastDeployedFiles"/> | ||||
|       <valuelist type="QVariantList" key="ProjectExplorer.RunConfiguration.LastDeployedHosts"/> | ||||
|       <valuelist type="QVariantList" key="ProjectExplorer.RunConfiguration.LastDeployedRemotePaths"/> | ||||
|       <valuelist type="QVariantList" key="ProjectExplorer.RunConfiguration.LastDeployedSysroots"/> | ||||
|       <valuelist type="QVariantList" key="RemoteLinux.LastDeployedLocalTimes"/> | ||||
|       <valuelist type="QVariantList" key="RemoteLinux.LastDeployedRemoteTimes"/> | ||||
|      </valuemap> | ||||
|      <value type="int" key="ProjectExplorer.BuildStepList.StepsCount">2</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DefaultDisplayName">Deploy</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Deploy</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">ProjectExplorer.BuildSteps.Deploy</value> | ||||
|     </valuemap> | ||||
|     <value type="int" key="ProjectExplorer.BuildConfiguration.BuildStepListCount">1</value> | ||||
|     <valuemap type="QVariantMap" key="ProjectExplorer.DeployConfiguration.CustomData"/> | ||||
|     <value type="bool" key="ProjectExplorer.DeployConfiguration.CustomDataEnabled">false</value> | ||||
|     <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">DeployToGenericLinux</value> | ||||
|    </valuemap> | ||||
|    <value type="int" key="ProjectExplorer.Target.DeployConfigurationCount">1</value> | ||||
|    <valuemap type="QVariantMap" key="ProjectExplorer.Target.RunConfiguration.0"> | ||||
|     <value type="bool" key="Analyzer.Perf.Settings.UseGlobalSettings">true</value> | ||||
|     <value type="bool" key="Analyzer.QmlProfiler.Settings.UseGlobalSettings">true</value> | ||||
|     <value type="bool" key="Analyzer.Valgrind.Settings.UseGlobalSettings">true</value> | ||||
|     <valuelist type="QVariantList" key="CustomOutputParsers"/> | ||||
|     <value type="int" key="PE.EnvironmentAspect.Base">1</value> | ||||
|     <valuelist type="QVariantList" key="PE.EnvironmentAspect.Changes"/> | ||||
|     <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">DownloadDCJsonFiles</value> | ||||
|     <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">RemoteLinuxRunConfiguration:/opt/ptu5/opt/DownloadDCJsonFiles/DownloadDCJsonFiles.pro</value> | ||||
|     <value type="QString" key="ProjectExplorer.RunConfiguration.BuildKey">/opt/ptu5/opt/DownloadDCJsonFiles/DownloadDCJsonFiles.pro</value> | ||||
|     <value type="int" key="RemoteLinux.EnvironmentAspect.Version">1</value> | ||||
|     <value type="bool" key="RunConfiguration.UseCppDebugger">false</value> | ||||
|     <value type="bool" key="RunConfiguration.UseCppDebuggerAuto">true</value> | ||||
|     <value type="bool" key="RunConfiguration.UseQmlDebugger">false</value> | ||||
|     <value type="bool" key="RunConfiguration.UseQmlDebuggerAuto">true</value> | ||||
|     <value type="QString" key="RunConfiguration.X11Forwarding">:0</value> | ||||
|    </valuemap> | ||||
|    <value type="int" key="ProjectExplorer.Target.RunConfigurationCount">1</value> | ||||
|   </valuemap> | ||||
|  </data> | ||||
|  <data> | ||||
|   <variable>ProjectExplorer.Project.Target.1</variable> | ||||
|   <valuemap type="QVariantMap"> | ||||
|    <value type="QString" key="DeviceType">Desktop</value> | ||||
|    <value type="QString" key="ProjectExplorer.ProjectConfiguration.DefaultDisplayName">Desktop Qt 5.12.12 GCC 64bit</value> | ||||
|    <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Desktop Qt 5.12.12 GCC 64bit</value> | ||||
|    <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">qt.qt5.51212.gcc_64_kit</value> | ||||
|    <value type="int" key="ProjectExplorer.Target.ActiveBuildConfiguration">0</value> | ||||
|    <value type="int" key="ProjectExplorer.Target.ActiveDeployConfiguration">0</value> | ||||
|    <value type="int" key="ProjectExplorer.Target.ActiveRunConfiguration">0</value> | ||||
|    <valuemap type="QVariantMap" key="ProjectExplorer.Target.BuildConfiguration.0"> | ||||
|     <value type="int" key="EnableQmlDebugging">0</value> | ||||
|     <value type="QString" key="ProjectExplorer.BuildConfiguration.BuildDirectory">/opt/ptu5/opt/build-DownloadDCJsonFiles-Desktop_Qt_5_12_12_GCC_64bit-Debug</value> | ||||
|     <value type="QString" key="ProjectExplorer.BuildConfiguration.BuildDirectory.shadowDir">/opt/ptu5/opt/build-DownloadDCJsonFiles-Desktop_Qt_5_12_12_GCC_64bit-Debug</value> | ||||
|     <valuemap type="QVariantMap" key="ProjectExplorer.BuildConfiguration.BuildStepList.0"> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.0"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">QtProjectManager.QMakeBuildStep</value> | ||||
|       <value type="bool" key="QtProjectManager.QMakeBuildStep.QMakeForced">false</value> | ||||
|       <valuelist type="QVariantList" key="QtProjectManager.QMakeBuildStep.SelectedAbis"/> | ||||
|      </valuemap> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.1"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">Qt4ProjectManager.MakeStep</value> | ||||
|      </valuemap> | ||||
|      <value type="int" key="ProjectExplorer.BuildStepList.StepsCount">2</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DefaultDisplayName">Build</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Build</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">ProjectExplorer.BuildSteps.Build</value> | ||||
|     </valuemap> | ||||
|     <valuemap type="QVariantMap" key="ProjectExplorer.BuildConfiguration.BuildStepList.1"> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.0"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">Qt4ProjectManager.MakeStep</value> | ||||
|       <value type="QString" key="Qt4ProjectManager.MakeStep.MakeArguments">clean</value> | ||||
|      </valuemap> | ||||
|      <value type="int" key="ProjectExplorer.BuildStepList.StepsCount">1</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DefaultDisplayName">Clean</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Clean</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">ProjectExplorer.BuildSteps.Clean</value> | ||||
|     </valuemap> | ||||
|     <value type="int" key="ProjectExplorer.BuildConfiguration.BuildStepListCount">2</value> | ||||
|     <value type="bool" key="ProjectExplorer.BuildConfiguration.ClearSystemEnvironment">false</value> | ||||
|     <valuelist type="QVariantList" key="ProjectExplorer.BuildConfiguration.CustomParsers"/> | ||||
|     <valuelist type="QVariantList" key="ProjectExplorer.BuildConfiguration.UserEnvironmentChanges"/> | ||||
|     <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Debug</value> | ||||
|     <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">Qt4ProjectManager.Qt4BuildConfiguration</value> | ||||
|     <value type="int" key="Qt4ProjectManager.Qt4BuildConfiguration.BuildConfiguration">2</value> | ||||
|    </valuemap> | ||||
|    <valuemap type="QVariantMap" key="ProjectExplorer.Target.BuildConfiguration.1"> | ||||
|     <value type="QString" key="ProjectExplorer.BuildConfiguration.BuildDirectory">/opt/ptu5/opt/build-DownloadDCJsonFiles-Desktop_Qt_5_12_12_GCC_64bit-Release</value> | ||||
|     <value type="QString" key="ProjectExplorer.BuildConfiguration.BuildDirectory.shadowDir">/opt/ptu5/opt/build-DownloadDCJsonFiles-Desktop_Qt_5_12_12_GCC_64bit-Release</value> | ||||
|     <valuemap type="QVariantMap" key="ProjectExplorer.BuildConfiguration.BuildStepList.0"> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.0"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">QtProjectManager.QMakeBuildStep</value> | ||||
|       <value type="bool" key="QtProjectManager.QMakeBuildStep.QMakeForced">false</value> | ||||
|       <valuelist type="QVariantList" key="QtProjectManager.QMakeBuildStep.SelectedAbis"/> | ||||
|      </valuemap> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.1"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">Qt4ProjectManager.MakeStep</value> | ||||
|      </valuemap> | ||||
|      <value type="int" key="ProjectExplorer.BuildStepList.StepsCount">2</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DefaultDisplayName">Build</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Build</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">ProjectExplorer.BuildSteps.Build</value> | ||||
|     </valuemap> | ||||
|     <valuemap type="QVariantMap" key="ProjectExplorer.BuildConfiguration.BuildStepList.1"> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.0"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">Qt4ProjectManager.MakeStep</value> | ||||
|       <value type="QString" key="Qt4ProjectManager.MakeStep.MakeArguments">clean</value> | ||||
|      </valuemap> | ||||
|      <value type="int" key="ProjectExplorer.BuildStepList.StepsCount">1</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DefaultDisplayName">Clean</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Clean</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">ProjectExplorer.BuildSteps.Clean</value> | ||||
|     </valuemap> | ||||
|     <value type="int" key="ProjectExplorer.BuildConfiguration.BuildStepListCount">2</value> | ||||
|     <value type="bool" key="ProjectExplorer.BuildConfiguration.ClearSystemEnvironment">false</value> | ||||
|     <valuelist type="QVariantList" key="ProjectExplorer.BuildConfiguration.CustomParsers"/> | ||||
|     <valuelist type="QVariantList" key="ProjectExplorer.BuildConfiguration.UserEnvironmentChanges"/> | ||||
|     <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Release</value> | ||||
|     <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">Qt4ProjectManager.Qt4BuildConfiguration</value> | ||||
|     <value type="int" key="Qt4ProjectManager.Qt4BuildConfiguration.BuildConfiguration">0</value> | ||||
|     <value type="int" key="QtQuickCompiler">0</value> | ||||
|    </valuemap> | ||||
|    <valuemap type="QVariantMap" key="ProjectExplorer.Target.BuildConfiguration.2"> | ||||
|     <value type="int" key="EnableQmlDebugging">0</value> | ||||
|     <value type="QString" key="ProjectExplorer.BuildConfiguration.BuildDirectory">/opt/ptu5/opt/build-DownloadDCJsonFiles-Desktop_Qt_5_12_12_GCC_64bit-Profile</value> | ||||
|     <value type="QString" key="ProjectExplorer.BuildConfiguration.BuildDirectory.shadowDir">/opt/ptu5/opt/build-DownloadDCJsonFiles-Desktop_Qt_5_12_12_GCC_64bit-Profile</value> | ||||
|     <valuemap type="QVariantMap" key="ProjectExplorer.BuildConfiguration.BuildStepList.0"> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.0"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">QtProjectManager.QMakeBuildStep</value> | ||||
|       <value type="bool" key="QtProjectManager.QMakeBuildStep.QMakeForced">false</value> | ||||
|       <valuelist type="QVariantList" key="QtProjectManager.QMakeBuildStep.SelectedAbis"/> | ||||
|      </valuemap> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.1"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">Qt4ProjectManager.MakeStep</value> | ||||
|      </valuemap> | ||||
|      <value type="int" key="ProjectExplorer.BuildStepList.StepsCount">2</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DefaultDisplayName">Build</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Build</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">ProjectExplorer.BuildSteps.Build</value> | ||||
|     </valuemap> | ||||
|     <valuemap type="QVariantMap" key="ProjectExplorer.BuildConfiguration.BuildStepList.1"> | ||||
|      <valuemap type="QVariantMap" key="ProjectExplorer.BuildStepList.Step.0"> | ||||
|       <value type="bool" key="ProjectExplorer.BuildStep.Enabled">true</value> | ||||
|       <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">Qt4ProjectManager.MakeStep</value> | ||||
|       <value type="QString" key="Qt4ProjectManager.MakeStep.MakeArguments">clean</value> | ||||
|      </valuemap> | ||||
|      <value type="int" key="ProjectExplorer.BuildStepList.StepsCount">1</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DefaultDisplayName">Clean</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Clean</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">ProjectExplorer.BuildSteps.Clean</value> | ||||
|     </valuemap> | ||||
|     <value type="int" key="ProjectExplorer.BuildConfiguration.BuildStepListCount">2</value> | ||||
|     <value type="bool" key="ProjectExplorer.BuildConfiguration.ClearSystemEnvironment">false</value> | ||||
|     <valuelist type="QVariantList" key="ProjectExplorer.BuildConfiguration.CustomParsers"/> | ||||
|     <valuelist type="QVariantList" key="ProjectExplorer.BuildConfiguration.UserEnvironmentChanges"/> | ||||
|     <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Profile</value> | ||||
|     <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">Qt4ProjectManager.Qt4BuildConfiguration</value> | ||||
|     <value type="int" key="Qt4ProjectManager.Qt4BuildConfiguration.BuildConfiguration">0</value> | ||||
|     <value type="int" key="QtQuickCompiler">0</value> | ||||
|     <value type="int" key="SeparateDebugInfo">0</value> | ||||
|    </valuemap> | ||||
|    <value type="int" key="ProjectExplorer.Target.BuildConfigurationCount">3</value> | ||||
|    <valuemap type="QVariantMap" key="ProjectExplorer.Target.DeployConfiguration.0"> | ||||
|     <valuemap type="QVariantMap" key="ProjectExplorer.BuildConfiguration.BuildStepList.0"> | ||||
|      <value type="int" key="ProjectExplorer.BuildStepList.StepsCount">0</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DefaultDisplayName">Deploy</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.DisplayName">Deploy</value> | ||||
|      <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">ProjectExplorer.BuildSteps.Deploy</value> | ||||
|     </valuemap> | ||||
|     <value type="int" key="ProjectExplorer.BuildConfiguration.BuildStepListCount">1</value> | ||||
|     <valuemap type="QVariantMap" key="ProjectExplorer.DeployConfiguration.CustomData"/> | ||||
|     <value type="bool" key="ProjectExplorer.DeployConfiguration.CustomDataEnabled">false</value> | ||||
|     <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">ProjectExplorer.DefaultDeployConfiguration</value> | ||||
|    </valuemap> | ||||
|    <value type="int" key="ProjectExplorer.Target.DeployConfigurationCount">1</value> | ||||
|    <valuemap type="QVariantMap" key="ProjectExplorer.Target.RunConfiguration.0"> | ||||
|     <value type="bool" key="Analyzer.Perf.Settings.UseGlobalSettings">true</value> | ||||
|     <value type="bool" key="Analyzer.QmlProfiler.Settings.UseGlobalSettings">true</value> | ||||
|     <value type="bool" key="Analyzer.Valgrind.Settings.UseGlobalSettings">true</value> | ||||
|     <valuelist type="QVariantList" key="CustomOutputParsers"/> | ||||
|     <value type="int" key="PE.EnvironmentAspect.Base">2</value> | ||||
|     <valuelist type="QVariantList" key="PE.EnvironmentAspect.Changes"/> | ||||
|     <value type="QString" key="ProjectExplorer.ProjectConfiguration.Id">ProjectExplorer.CustomExecutableRunConfiguration</value> | ||||
|     <value type="QString" key="ProjectExplorer.RunConfiguration.BuildKey"></value> | ||||
|     <value type="bool" key="RunConfiguration.UseCppDebugger">false</value> | ||||
|     <value type="bool" key="RunConfiguration.UseCppDebuggerAuto">true</value> | ||||
|     <value type="bool" key="RunConfiguration.UseQmlDebugger">false</value> | ||||
|     <value type="bool" key="RunConfiguration.UseQmlDebuggerAuto">true</value> | ||||
|    </valuemap> | ||||
|    <value type="int" key="ProjectExplorer.Target.RunConfigurationCount">1</value> | ||||
|   </valuemap> | ||||
|  </data> | ||||
|  <data> | ||||
|   <variable>ProjectExplorer.Project.TargetCount</variable> | ||||
|   <value type="int">2</value> | ||||
|  </data> | ||||
|  <data> | ||||
|   <variable>ProjectExplorer.Project.Updater.FileVersion</variable> | ||||
|   <value type="int">22</value> | ||||
|  </data> | ||||
|  <data> | ||||
|   <variable>Version</variable> | ||||
|   <value type="int">22</value> | ||||
|  </data> | ||||
| </qtcreator> | ||||
							
								
								
									
										0
									
								
								DownloadDCJsonFiles/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								DownloadDCJsonFiles/README.md
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										145
									
								
								DownloadDCJsonFiles/main.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										145
									
								
								DownloadDCJsonFiles/main.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,145 @@ | ||||
|  | ||||
| #include <QtGlobal> | ||||
| #include <QCoreApplication> | ||||
| #include <QByteArray> | ||||
|  | ||||
|  | ||||
| #include <QProcess> | ||||
| #include <QCommandLineParser> | ||||
| #include <QStandardPaths> | ||||
| #include <QSettings> | ||||
| #include <QDir> | ||||
| #include <QDebug> | ||||
|  | ||||
| #include "message_handler.h" | ||||
| #include "commandline_parser.h" | ||||
| #include "utils.h" | ||||
| #include "update.h" | ||||
|  | ||||
| #include <DeviceController/interfaces.h> | ||||
|  | ||||
|  | ||||
| //#include <unistd.h> | ||||
| //#include <errno.h> | ||||
|  | ||||
| #ifdef PTU5 | ||||
| #define SERIAL_PORT "ttymxc2" | ||||
| #else | ||||
| #define SERIAL_PORT "ttyUSB0" | ||||
| #endif | ||||
|  | ||||
| int read1stLineOfFile(QString fileName) { | ||||
|     QFile f(fileName); | ||||
|     if (f.exists()) { | ||||
|         if (f.open(QIODevice::ReadOnly | QIODevice::Text)) { | ||||
|             QTextStream in(&f); | ||||
|             in.setCodec("UTF-8"); | ||||
|             while(!in.atEnd()) { | ||||
|                 return in.readLine().toInt(); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     return -1; | ||||
| } | ||||
|  | ||||
| int main(int argc, char **argv) { | ||||
|     QByteArray const value = qgetenv("LC_ALL"); | ||||
|     if (value != "C") { | ||||
|         qputenv("LC_ALL", "C"); | ||||
|     } | ||||
|     // qputenv("XDG_RUNTIME_DIR", "/var/run/user/0"); | ||||
|  | ||||
|     openlog("ATB-DL-JSON", LOG_PERROR | LOG_PID | LOG_CONS, LOG_USER); | ||||
|  | ||||
|     QCoreApplication a(argc, argv); | ||||
|     QCoreApplication::setApplicationName("ATBDownloadDCJsonFiles"); | ||||
|     QCoreApplication::setApplicationVersion(APP_VERSION); | ||||
|  | ||||
|     if (!messageHandlerInstalled()) { // change internal qt-QDebug-handling | ||||
|         atbInstallMessageHandler(atbDebugOutput); | ||||
|         setDebugLevel(LOG_NOTICE); | ||||
|     } | ||||
|  | ||||
|     CommandLineParser parser; | ||||
|     parser.process(a); | ||||
|     parser.readSettings(); | ||||
|  | ||||
|     QString repositoryUrl = parser.repositoryUrl(); | ||||
|     QString plugInDir = parser.plugInDir(); | ||||
|     QString plugInName = parser.plugInName(); | ||||
|     QString workingDir = parser.workingDir(); | ||||
|     QString iniFileName = parser.iniFileName(); | ||||
|     bool const dryRun = parser.dryRun(); | ||||
|     bool const noUpdatePsaHardware = parser.noUpdatePsaHardware(); | ||||
|     bool const showYoctoVersion = parser.yoctoVersion(); | ||||
|     bool const showYoctoInstallStatus = parser.yoctoInstallStatus(); | ||||
|     bool const showExtendedVersion = parser.extendedVersion(); | ||||
|     bool const alwaysDownloadConfig = parser.alwaysDownloadConfig(); | ||||
|     bool const alwaysDownloadDC = parser.alwaysDownloadDC(); | ||||
|  | ||||
|     QString const rtPath = QCoreApplication::applicationDirPath(); | ||||
|  | ||||
|     int const machineNr = read1stLineOfFile("/mnt/system_data/machine_nr"); | ||||
|     int const customerNr = read1stLineOfFile("/mnt/system_data/cust_nr"); | ||||
|     int const zoneNr = read1stLineOfFile("/mnt/system_data/zone_nr"); | ||||
|     QString const branchName = (zoneNr != 0) | ||||
|             ? QString("zg1/zone%1").arg(zoneNr) : "master"; | ||||
|  | ||||
|     qInfo() << "pwd ......................" << rtPath; | ||||
|     qInfo() << "repositoryUrl ............" << repositoryUrl; | ||||
|     qInfo() << "plugInDir ................" << plugInDir; | ||||
|     qInfo() << "plugInName ..............." << plugInName; | ||||
|     qInfo() << "workingDir ..............." << workingDir; | ||||
|     qInfo() << "dryRun ..................." << dryRun; | ||||
|     qInfo() << "noUpdatePsaHardware ......" << noUpdatePsaHardware; | ||||
|     qInfo() << "alwaysDownloadConfig ....." << alwaysDownloadConfig; | ||||
|     qInfo() << "alwaysDownloadDC ........." << alwaysDownloadDC; | ||||
|     qInfo() << "showYoctoVersion ........." << showYoctoVersion; | ||||
|     qInfo() << "showYoctoInstallStatus ..." << showYoctoInstallStatus; | ||||
|     qInfo() << "showExtendedVersion ......" << showExtendedVersion; | ||||
|     qInfo() << "iniFileName .............." << iniFileName; | ||||
|     qInfo() << "extended-version ........." << APP_EXTENDED_VERSION; | ||||
|     qInfo() << "machineNr ................" << machineNr; | ||||
|     qInfo() << "customerNr ..............." << customerNr; | ||||
|     qInfo() << "zoneNr ..................." << zoneNr; | ||||
|  | ||||
|     if (showExtendedVersion) { | ||||
|         printf(APP_EXTENDED_VERSION"\n"); | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     QString const customerRepo = QDir::cleanPath(workingDir + QDir::separator() + QString("customer_%1").arg(customerNr)); | ||||
|  | ||||
|     qCritical() << "Using customer repository" << customerRepo; | ||||
|  | ||||
|     // always execute contents of opkg_commands-file | ||||
|     QStringList filesToUpdate; | ||||
|  | ||||
|     QDir dir(QDir::cleanPath(customerRepo + QDir::separator() + "etc/psa_config")); | ||||
|  | ||||
|     if (dir.exists()) { | ||||
|         QStringList jsons = dir.entryList(QStringList() << "DC2C*.json", QDir::Files); | ||||
|         if (!jsons.isEmpty()) { | ||||
|             for (QStringList::size_type i=0; i<jsons.size(); ++i) { | ||||
|                 filesToUpdate << QDir::cleanPath(QString("etc/psa_config/") + jsons.at(i)); | ||||
|             } | ||||
|         } | ||||
|     } else { | ||||
|         qCritical() << "DIRECTORY" << dir << "DOES NOT EXIST"; | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     qCritical() << "JSON FILES TO UPDATE" << filesToUpdate; | ||||
|  | ||||
|     Update update(customerRepo, | ||||
|                   QString::number(customerNr), | ||||
|                   branchName, | ||||
|                   plugInDir, | ||||
|                   plugInName, | ||||
|                   workingDir); | ||||
|  | ||||
|     update.doUpdate(filesToUpdate); | ||||
|  | ||||
|     //return a.exec(); | ||||
|     return 0; | ||||
| } | ||||
							
								
								
									
										97
									
								
								DownloadDCJsonFiles/message_handler.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										97
									
								
								DownloadDCJsonFiles/message_handler.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,97 @@ | ||||
| #include "message_handler.h" | ||||
|  | ||||
| #include <QDateTime> | ||||
| #include <cstring> | ||||
| #include <QString> | ||||
| #include <QFileInfo> | ||||
| #include <QMessageLogContext> | ||||
|  | ||||
|  | ||||
| static char const *DBG_NAME[] = { "DBG  ", "WARN ", "CRIT ", "FATAL", "INFO " }; | ||||
| static bool installedMsgHandler = false; | ||||
| static int debugLevel = LOG_NOTICE; | ||||
|  | ||||
| int getDebugLevel() { return debugLevel; } | ||||
| void setDebugLevel(int newDebugLevel) { | ||||
|     debugLevel = newDebugLevel; | ||||
| } | ||||
|  | ||||
| bool messageHandlerInstalled() { | ||||
|     return installedMsgHandler; | ||||
| } | ||||
|  | ||||
| QtMessageHandler atbInstallMessageHandler(QtMessageHandler handler) { | ||||
|     installedMsgHandler = (handler != 0); | ||||
|     static QtMessageHandler prevHandler = nullptr; | ||||
|     if (handler) { | ||||
|         prevHandler = qInstallMessageHandler(handler); | ||||
|         return prevHandler; | ||||
|     } else { | ||||
|         return qInstallMessageHandler(prevHandler); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// | ||||
| /// \brief Print message according to given debug level. | ||||
| /// | ||||
| /// \note Install this function using qInstallMsgHandler(). | ||||
| /// | ||||
| ///  int main(int argc, char **argv) { | ||||
| ///     installMsgHandler(atbDebugOutput); | ||||
| ///     QApplication app(argc, argv); | ||||
| ///     ... | ||||
| ///     return app.exec(); | ||||
| ///  } | ||||
| /// | ||||
| #if (QT_VERSION > QT_VERSION_CHECK(5, 0, 0) && QT_VERSION < QT_VERSION_CHECK(6, 0, 0)) | ||||
| void atbDebugOutput(QtMsgType type, const QMessageLogContext &context, const QString &msg) { | ||||
|     Q_UNUSED(context); | ||||
|     QString const localMsg = QString(DBG_NAME[type]) + msg.toLocal8Bit(); | ||||
|  | ||||
|     switch (debugLevel) { | ||||
|         case LOG_DEBUG: { // debug-level message | ||||
|             syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|         } break; | ||||
|         case LOG_INFO: { // informational message | ||||
|             if (type != QtDebugMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         case LOG_NOTICE: { // normal, but significant, condition | ||||
|             if (type != QtDebugMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         case LOG_WARNING: { // warning conditions | ||||
|             if (type != QtInfoMsg && type != QtDebugMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         case LOG_ERR: { // error conditions | ||||
|             if (type != QtInfoMsg && type != QtDebugMsg && type != QtWarningMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         case LOG_CRIT: { // critical conditions | ||||
|             if (type != QtInfoMsg && type != QtDebugMsg && type != QtWarningMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         case LOG_ALERT: { // action must be taken immediately | ||||
|             if (type != QtInfoMsg && type != QtDebugMsg && type != QtWarningMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         case LOG_EMERG: { // system is unusable | ||||
|             if (type != QtInfoMsg && type != QtDebugMsg && type != QtWarningMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         default: { | ||||
|             //fprintf(stderr, "%s No ErrorLevel defined! %s\n", | ||||
|             //    datetime.toStdString().c_str(), msg.toStdString().c_str()); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| #endif | ||||
|  | ||||
							
								
								
									
										23
									
								
								DownloadDCJsonFiles/message_handler.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								DownloadDCJsonFiles/message_handler.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| #ifndef MESSAGE_HANDLER_H_INCLUDED | ||||
| #define MESSAGE_HANDLER_H_INCLUDED | ||||
|  | ||||
| #include <QtGlobal> | ||||
| #ifdef __linux__ | ||||
| #include <syslog.h> | ||||
| #endif | ||||
|  | ||||
| int getDebugLevel(); | ||||
| void setDebugLevel(int newDebugLevel); | ||||
|  | ||||
| bool messageHandlerInstalled(); | ||||
| QtMessageHandler atbInstallMessageHandler(QtMessageHandler handler); | ||||
|  | ||||
| #if QT_VERSION < QT_VERSION_CHECK(5, 0, 0) | ||||
| // typedef void (*QtMessageHandler)(QtMsgType, const char *); | ||||
| void atbDebugOutput(QtMsgType type, const char *msg); | ||||
| #elif QT_VERSION < QT_VERSION_CHECK(6, 0, 0) | ||||
| // typedef void (*QtMessageHandler)(QtMsgType, const QMessageLogContext &, const QString &); | ||||
| void atbDebugOutput(QtMsgType type, const QMessageLogContext &context, const QString &msg); | ||||
| #endif | ||||
|  | ||||
| #endif // MESSAGE_HANDLER_H_INCLUDED | ||||
							
								
								
									
										128
									
								
								plugins/interfaces.h → DownloadDCJsonFiles/plugins/interfaces.h
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							
							
						
						
									
										128
									
								
								plugins/interfaces.h → DownloadDCJsonFiles/plugins/interfaces.h
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							| @@ -24,6 +24,7 @@ struct T_emp | ||||
| 
 | ||||
|    // dynamic:
 | ||||
|    uint8_t state;      // step counter of EMP (electronic coin checker) FSM (finite state machine):
 | ||||
|    /*
 | ||||
|    // 0=start command
 | ||||
|        // 1=powered, do emp ini, send reset
 | ||||
|        // 2=delay
 | ||||
| @@ -39,7 +40,7 @@ struct T_emp | ||||
|        // 90: stop all, 1s delay
 | ||||
|        // 99: off, all stopped
 | ||||
| 
 | ||||
| 
 | ||||
| */ | ||||
|    uint8_t pollingRunning; | ||||
|    uint8_t paymentRunning; | ||||
| 
 | ||||
| @@ -238,6 +239,7 @@ struct T_dynamicCondition | ||||
|         uint8_t lastVDoorState; | ||||
|         uint8_t lastCBstate; | ||||
|         char        paymentInProgress; | ||||
|         // Version Szeged:  aug2023
 | ||||
|         //	0: stopped by timeout
 | ||||
|         //  1: running  2: wait4lastCoin
 | ||||
|         //				3: payment stopped manually, coins in Escrow
 | ||||
| @@ -246,6 +248,21 @@ struct T_dynamicCondition | ||||
|         //			    6: coins encashed   7:coins returned
 | ||||
|         //				8: CoinChecker or MDB on Error
 | ||||
| 
 | ||||
|         // since Schoenau with bill and changer, nov2023
 | ||||
|         //0 = no payment
 | ||||
|         //will be set to 1 by cash_startCollection()
 | ||||
|         //neu 1: wait for devices getting ready for payment
 | ||||
|         //2 = payment,
 | ||||
|         //3 = wait for last coin/bill
 | ||||
|         //4 = Bezahlvorgang manuell beendet
 | ||||
|         //5 = payment stopped autom, amount collected, coins in Escrow
 | ||||
|         //6 = Bezahlvorgang beendet weil ZK voll
 | ||||
|         //4,5,6: payment done, keep on polling, wait for cash or return command
 | ||||
|         //7 = encash collected money from coin escrow into cash box
 | ||||
|         //8 = return "amountToReturn", can be complete inserted amount or only overpayment
 | ||||
|         //9 = wait for changer result
 | ||||
|         //10= print refund receipt with "amountToReturn"
 | ||||
| 
 | ||||
| 
 | ||||
|         char        res1; | ||||
|         uint16_t    U_Batt; | ||||
| @@ -397,6 +414,7 @@ struct T_bna | ||||
| 
 | ||||
| }; | ||||
| 
 | ||||
| class DownloadThread; | ||||
| class hwinf | ||||
| { | ||||
| 
 | ||||
| @@ -1834,6 +1852,9 @@ public: | ||||
|     virtual uint8_t prn_getPrintResult() const { | ||||
|        return 0; | ||||
|     } | ||||
|     // return:  0: just printing, wait
 | ||||
|     //          1: OK - last print was succesful
 | ||||
|     //          2: error - not printed
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| @@ -2249,14 +2270,23 @@ public: | ||||
|     virtual void bna_requestCurrentNotes() const {} | ||||
|     // send command to DC in order to get transaction data
 | ||||
| 
 | ||||
|     virtual uint8_t bna_getCurrentNotes(uint16_t latestBill, uint16_t *currentNotes) const { | ||||
|     virtual uint8_t bna_getCurrentNotes(uint16_t latestBill, uint16_t *currentNotes) const | ||||
|     { | ||||
|         Q_UNUSED(latestBill); | ||||
|         Q_UNUSED(currentNotes); | ||||
|         return 0; | ||||
|     } | ||||
|     // returns number of collected bank notes since start-command (current transaction)
 | ||||
|     //  latestBill: last accepted bank note, value in cent
 | ||||
|     //  currentNotes an array with up to 16 (further) notes collected
 | ||||
|     // return value: numbers of bills or 99 in case of error
 | ||||
|     //  latestBill: not used
 | ||||
|     // in case of error: currentNotes[0,1,2,3] = 1..4 error number(s)
 | ||||
|     // in normal case:
 | ||||
|     //  currentNotes[0]: last bill in cent (e.g. 1000 = 10€)
 | ||||
|     //  currentNotes[1]: bin 1 = bill is still in escrow   else bill is stacked
 | ||||
|     //       note: by now (dec2023) escrow is not used, bills always go to stacker (box)
 | ||||
|     //  currentNotes[2]: total sum of bills in cent, low word (16bit)
 | ||||
|     //  currentNotes[3]: total sum of bills in cent, high word (16bit)
 | ||||
| 
 | ||||
| 
 | ||||
|     virtual void bna_requestStackerLevel() const {} | ||||
| 
 | ||||
| @@ -2270,10 +2300,99 @@ public: | ||||
|     //                                          countOfBills[1] for 10€   and so on
 | ||||
| 
 | ||||
| 
 | ||||
|     // download device controller
 | ||||
|     virtual bool dcDownloadRequest(QString const &fileToDownload) const { | ||||
|         Q_UNUSED(fileToDownload); | ||||
|         return false; | ||||
|     } | ||||
|     virtual bool dcDownloadRequested() const { return false; } | ||||
|     virtual bool dcDownloadResetRequest() const { return false; } | ||||
|     virtual bool dcDownloadRequestAck() const { return false; } | ||||
|     virtual bool dcDownloadRunning() const { return false; } | ||||
|     virtual bool dcDownloadFinished() { return false; } | ||||
| 
 | ||||
|     virtual bool dcDownloadReportStart() const { return false; } | ||||
|     virtual bool dcDownloadReportRunning() const { return true; } | ||||
|     virtual bool dcDownloadReportFinished() { return true; } | ||||
| 
 | ||||
|     virtual bool dcDownloadThreadStart() { return false; } | ||||
|     virtual bool dcDownloadThreadRunning() const { return true; } | ||||
|     virtual void dcDownloadThreadFinalize(DownloadThread *) {} | ||||
|     virtual bool dcDownloadThreadFinished() const { return true; } | ||||
|     virtual bool dcDownloadReportThreadStart() { return false; } | ||||
|     virtual bool dcDownloadReportThreadRunning() const { return true; } | ||||
|     virtual void dcDownloadReportThreadFinalize() {} | ||||
|     virtual void dcDownloadReportThreadQuit() {} | ||||
|     virtual bool dcDownloadReportThreadFinished() const { return true; } | ||||
| 
 | ||||
| 
 | ||||
|     virtual QString dcDownloadFileName() const { return ""; } | ||||
|     virtual bool dcDownloadSetRequested(bool requested) { | ||||
|        Q_UNUSED(requested); return false; | ||||
|     } | ||||
|     virtual bool dcDownloadSetRunning(bool running) { | ||||
|        Q_UNUSED(running); return false; | ||||
|     } | ||||
|     virtual bool dcDownloadSetFinished(bool finished) { | ||||
|        Q_UNUSED(finished); return false; | ||||
|     } | ||||
|     virtual void dcDownloadSetTotalBlockNumber(uint16_t totalBlockNumber) { | ||||
|        Q_UNUSED(totalBlockNumber); | ||||
|     } | ||||
|     virtual void dcDownloadSetCurrentBlockNumber(uint16_t currentBlockNumber) { | ||||
|        Q_UNUSED(currentBlockNumber); | ||||
|     } | ||||
|     virtual bool dcDownloadGetRequested() const { return false; } | ||||
|     virtual bool dcDownloadGetRunning() const { return false; } | ||||
|     virtual bool dcDownloadGetFinished() const { return false; } | ||||
|     virtual uint16_t dcDownloadGetTotalBlockNumber() const { return 0; } | ||||
|     virtual uint16_t dcDownloadGetCurrentBlockNumber() const { return 0; } | ||||
| 
 | ||||
|     virtual QObject const *getAPI() { return nullptr; } | ||||
| 
 | ||||
| 
 | ||||
| signals: | ||||
|     /*
 | ||||
|         NOTE: the difference between a virtual Qt signal and a normal Qt signal: | ||||
|         A Qt virtual signal is a connection that is established using a pointer | ||||
|         or reference and is not connected to an object or data. It is therefore | ||||
|         not bound to a particular object, but to a specific class (object type). | ||||
|         Qt virtual signals are useful because they allow you to create | ||||
|         connections without worrying about whether an object or a specific data | ||||
|         element has been destroyed. | ||||
|         https://www.youtube.com/watch?v=HTH3VFfqsXw
 | ||||
| 
 | ||||
|      */ | ||||
|     virtual void hwapi_reportDCDownloadStatus(QString const&) const {} | ||||
|     virtual void hwapi_reportDCDownloadSuccess(QString const&) const {} | ||||
|     virtual void hwapi_reportDCDownloadFailure(QString const&) const {} | ||||
| 
 | ||||
|     virtual void hwapi_templatePrintFinished_OK(void) const=0; | ||||
|     virtual void hwapi_templatePrintFinished_Err(void) const=0; | ||||
| 
 | ||||
|     virtual void hwapi_coinCollectionJustStarted(void) const=0; | ||||
|     virtual void hwapi_coinCollectionAborted(void) const=0; | ||||
| 
 | ||||
|     virtual void hwapi_gotNewCoin(void) const=0; | ||||
|     virtual void hwapi_payStopByMax(void) const=0; | ||||
|     virtual void hwapi_payStopByPushbutton(void) const=0; | ||||
| 
 | ||||
|     virtual void hwapi_payStopByEscrow(void) const=0; | ||||
|     virtual void hwapi_payStopByError(void) const=0; | ||||
|     virtual void hwapi_payStopByTimeout(void) const=0; | ||||
|     virtual void hwapi_payCancelled(void) const=0; | ||||
|     virtual void hwapi_coinProcessJustStopped(void) const=0; | ||||
| 
 | ||||
|     virtual void hwapi_doorServiceDoorOpened(void) const=0; | ||||
|     virtual void hwapi_doorVaultDoorOpened(void) const=0; | ||||
|     virtual void hwapi_doorCoinBoxRemoved(void) const=0; | ||||
|     virtual void hwapi_doorCoinBoxInserted(void) const=0; | ||||
|     virtual void hwapi_doorCBinAndAllDoorsClosed(void) const=0; | ||||
|     virtual void hwapi_doorAllDoorsClosed(void) const=0; | ||||
| 
 | ||||
| 
 | ||||
|     // NOTE: declaring a "pure virtual" "signal" should be an error and thus not valid.
 | ||||
|     /* GH Version, bringt Fehler
 | ||||
|     void hwapi_templatePrintFinished_OK() const; | ||||
|     void hwapi_templatePrintFinished_Err() const; | ||||
| 
 | ||||
| @@ -2296,6 +2415,7 @@ signals: | ||||
|     void hwapi_doorCoinBoxInserted() const; | ||||
|     void hwapi_doorCBinAndAllDoorsClosed() const; | ||||
|     void hwapi_doorAllDoorsClosed() const; | ||||
|     */ | ||||
| }; | ||||
| 
 | ||||
| 
 | ||||
							
								
								
									
										36
									
								
								DownloadDCJsonFiles/process/command.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								DownloadDCJsonFiles/process/command.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| #ifndef COMMAND_H_INCLUDED | ||||
| #define COMMAND_H_INCLUDED | ||||
|  | ||||
| #include <QObject> | ||||
| #include <QCoreApplication> | ||||
| #include <QString> | ||||
| #include <QStringList> | ||||
| #include <QProcess> | ||||
|  | ||||
|  | ||||
| class Command : public QObject { | ||||
|     Q_OBJECT | ||||
|  | ||||
|     QString m_command; | ||||
|     QString m_commandResult; | ||||
|     int m_waitForStartTimeout; | ||||
|     int m_waitForFinishTimeout; | ||||
|     int m_exitCode; | ||||
| public: | ||||
|     explicit Command(QString const &command, | ||||
|                      int start_timeout = 100000, | ||||
|                      int finish_timeout = 100000); | ||||
|  | ||||
|     QString getCommandResult() const; | ||||
|     QString command() const { return m_command; } | ||||
|  | ||||
|     bool execute(QString workingDirectory, QStringList args = QStringList()); | ||||
|     int exitCode() const { return m_exitCode; } | ||||
|  | ||||
| private slots: | ||||
|     void readyReadStandardOutput(); | ||||
|     void readyReadStandardError(); | ||||
|     void finished(int exitCode, QProcess::ExitStatus exitStatus); | ||||
| }; | ||||
|  | ||||
| #endif // COMMAND_H_INCLUDED | ||||
							
								
								
									
										390
									
								
								DownloadDCJsonFiles/update.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										390
									
								
								DownloadDCJsonFiles/update.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,390 @@ | ||||
| #include "update.h" | ||||
|  | ||||
| #include <QCoreApplication> | ||||
| #include <QFile> | ||||
| #include <QTemporaryFile> | ||||
| #include <QDebug> | ||||
| #include <QTextStream> | ||||
| #include <QRegularExpression> | ||||
| #include <QRegExp> | ||||
|  | ||||
| #if defined (Q_OS_UNIX) || defined (Q_OS_LINUX) | ||||
| #include "unistd.h" | ||||
| #endif | ||||
|  | ||||
| #include <DeviceController/interfaces.h> | ||||
|  | ||||
| #include <QSharedMemory> | ||||
| #include <QScopedPointer> | ||||
| #include <QDir> | ||||
| #include <QThread> | ||||
| #include <QDateTime> | ||||
| #include <QPluginLoader> | ||||
| #include <QMap> | ||||
|  | ||||
| #define UPDATE_OPKG                 (1) | ||||
| #define UPDATE_DC                   (0) | ||||
|  | ||||
| static const QMap<QString, int> baudrateMap = { | ||||
|   {"1200"   ,   0}, {"9600"   ,   1}, {"19200"  ,   2}, {"38400"  ,   3}, | ||||
|   {"57600"  ,   4}, {"115200" ,   5} | ||||
| }; | ||||
|  | ||||
| QPluginLoader Update::pluginLoader; | ||||
|  | ||||
| hwinf *Update::loadDCPlugin(QDir const &plugInDir, QString const &fname) { | ||||
|     hwinf *hw = nullptr; | ||||
|     if (plugInDir.exists()) { | ||||
|         QString pluginLibName(fname); | ||||
|         pluginLibName = plugInDir.absoluteFilePath(pluginLibName); | ||||
|         QFileInfo info(pluginLibName); | ||||
|         if (info.exists()) { | ||||
|             pluginLibName = plugInDir.absoluteFilePath(pluginLibName); | ||||
|             pluginLoader.setFileName(pluginLibName); | ||||
|             // static QPluginLoader pluginLoader(pluginLibName); | ||||
|             if (!pluginLoader.load()) { | ||||
|                 qCritical() << "in directory" << plugInDir.absolutePath(); | ||||
|                 qCritical() << "cannot load plugin" << pluginLoader.fileName(); | ||||
|                 qCritical() << pluginLoader.errorString(); | ||||
|                 exit(-1); | ||||
|             } | ||||
|  | ||||
|             qCritical() << "loadDCPlugin() plugin directory:" << plugInDir.absolutePath(); | ||||
|             qCritical() << "loadDCPlugin() plugin file name:" << pluginLoader.fileName(); | ||||
|  | ||||
|             if (!pluginLoader.isLoaded()) { | ||||
|                 qCritical() << pluginLoader.errorString(); | ||||
|                 exit(-2); | ||||
|             } | ||||
|             QObject *plugin = pluginLoader.instance(); | ||||
|             if (!plugin) { | ||||
|                 qCritical() << "cannot start instance"; | ||||
|                 exit(-3); | ||||
|             } | ||||
|             if (! (hw = qobject_cast<hwinf *>(plugin))) { | ||||
|                 qCritical() << "cannot cast plugin" << plugin << "to hwinf"; | ||||
|                 exit(-4); | ||||
|             } | ||||
|         } else { | ||||
|             qCritical() << pluginLibName << "does not exist"; | ||||
|             exit(-5); | ||||
|         } | ||||
|     } else { | ||||
|         qCritical() << "plugins directory" << plugInDir.absolutePath() | ||||
|                     << "does not exist"; | ||||
|         exit(-6); | ||||
|     } | ||||
|     return hw; | ||||
| } | ||||
|  | ||||
| bool Update::unloadDCPlugin() { | ||||
|     if (pluginLoader.unload()) { | ||||
|         qCritical() << "unloaded plugin" << pluginLoader.fileName(); | ||||
|         // Note: will re-instantiate the library ! | ||||
|         // QObject *rootObject = pluginLoader.instance(); | ||||
|         // if (rootObject) { | ||||
|         //    qCritical() << "reloaded plugin: root object again available"; | ||||
|         //    return false; | ||||
|         // } | ||||
|         // qCritical()unloaded plugin: root object gone"; | ||||
|         return true; | ||||
|     } | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| class hwapi; | ||||
| Update::Update(QString customerRepository, | ||||
|                QString customerNrStr, | ||||
|                QString branchName, | ||||
|                QString plugInDir, | ||||
|                QString pluginName, | ||||
|                QString workingDir, | ||||
|                bool dryRun, | ||||
|                QObject *parent, | ||||
|                char const *serialInterface, | ||||
|                char const *baudrate) | ||||
|     : QObject(parent) | ||||
|     , m_hw(loadDCPlugin(QDir(plugInDir), pluginName)) | ||||
|     , m_serialInterface(serialInterface) | ||||
|     , m_baudrate(baudrate) | ||||
|     , m_customerRepository(customerRepository) | ||||
|     , m_customerNrStr(customerNrStr) | ||||
|     , m_branchName(branchName) | ||||
|     , m_pluginName(pluginName) | ||||
|     , m_workingDir(workingDir) | ||||
|     , m_dryRun(dryRun) | ||||
|     , m_sys_areDCdataValid(false) { | ||||
|  | ||||
|     if (!m_hw) { | ||||
|         qCritical() << "(" << __func__ << ":" << __LINE__ << ") m_hw == nullptr -> ca-slave plugin loaded ???"; | ||||
|     } else { | ||||
|         int tries = 20; | ||||
|         while ((m_sys_areDCdataValid = m_hw->sys_areDCdataValid()) == false) { | ||||
|             // must deliver 'true', only then are all data from hwapi valid | ||||
|             if (--tries < 0) { | ||||
|                 qCritical() << "ERROR!!! DC DATA NOT VALID -> CA-MASTER-PLUGIN NOT CONNECTED"; | ||||
|                 break; | ||||
|             } | ||||
|             m_hw->dc_autoRequest(true); | ||||
|             QThread::msleep(500); | ||||
|         } | ||||
|  | ||||
|         qCritical() << "(" << __func__ << ":" << __LINE__ << ") m_sys_areDCDataValid ..." | ||||
|                     << m_sys_areDCdataValid; | ||||
|  | ||||
| #if 0 | ||||
|         QObject const *obj = m_hw->getAPI(); | ||||
|         Q_ASSERT(obj != nullptr); | ||||
|  | ||||
|         QDebug critical = qCritical(); | ||||
|         critical << "connect() to onReportDCDownloadStatus() ..."; | ||||
|         if (!connect(obj, | ||||
|                      SIGNAL(hwapi_reportDCDownloadStatus(QString const&)), | ||||
|                      this, | ||||
|                      SLOT(onReportDCDownloadStatus(QString const &)))) { | ||||
|             critical << "FAILED"; | ||||
|         } else critical << "DONE"; | ||||
|  | ||||
|         critical = qCritical(); | ||||
|         critical << "connect() to onReportDCDownloadSuccess() ..."; | ||||
|         if (!connect(obj, | ||||
|             SIGNAL(hwapi_reportDCDownloadSuccess(QString const&)), this, | ||||
|             SLOT(onReportDCDownloadSuccess(QString const &)))) { | ||||
|             critical << "FAILED"; | ||||
|         } else critical << "DONE"; | ||||
|  | ||||
|         critical = qCritical(); | ||||
|         critical << "connect() to onReportDCDownloadFailure() ..."; | ||||
|         if (!connect(obj, | ||||
|             SIGNAL(hwapi_reportDCDownloadFailure(QString const &)), this, | ||||
|             SLOT(onReportDCDownloadFailure(QString const &)))) { | ||||
|             critical << "FAILED"; | ||||
|         } else critical << "DONE"; | ||||
| #endif | ||||
|     } | ||||
| } | ||||
|  | ||||
| Update::~Update() { | ||||
|     // unloadDCPlugin(); | ||||
| } | ||||
|  | ||||
| bool Update::doUpdate(QStringList const &filesToWorkOn) { | ||||
|  | ||||
|     int tries = 20; | ||||
|     while ((m_sys_areDCdataValid = m_hw->sys_areDCdataValid()) == false) { | ||||
|         // must deliver 'true', only then are all data from hwapi valid | ||||
|         if (--tries < 0) { | ||||
|             qCritical() << "ERROR!!! DC DATA NOT VALID -> CA-SLAVE-PLUGIN NOT CONNECTED"; | ||||
|             return false; | ||||
|         } | ||||
|         qCritical() << "ERROR!!! DC DATA NOT VALID -> CA-SLAVE-PLUGIN NOT CONNECTED (" << tries << ")"; | ||||
|         m_hw->dc_autoRequest(true); | ||||
|         QThread::msleep(500); | ||||
|     } | ||||
|  | ||||
|     bool res = false; | ||||
|  | ||||
|     QList<QString>::const_iterator it; | ||||
|     for (it = filesToWorkOn.cbegin(); it != filesToWorkOn.cend(); ++it) { | ||||
|         QString const &fToWorkOn = QDir::cleanPath(m_customerRepository + QDir::separator() + it->trimmed()); | ||||
|         if (fToWorkOn.contains("DC2C_print", Qt::CaseInsensitive) | ||||
|          && fToWorkOn.endsWith(".json", Qt::CaseInsensitive)) { | ||||
|             res = true; | ||||
|             int i = fToWorkOn.indexOf("DC2C_print", Qt::CaseInsensitive); | ||||
|             int const templateIdx = fToWorkOn.mid(i).midRef(10, 2).toInt(); | ||||
|             if ((templateIdx < 1) || (templateIdx > 32)) { | ||||
|                 qCritical() << "WRONG TEMPLATE INDEX" << templateIdx; | ||||
|                 res = false; | ||||
|             } else { | ||||
|                 if ((res = updatePrinterTemplate(templateIdx, fToWorkOn))) { | ||||
|                     qCritical() << | ||||
|                         QString("DOWNLOADED PRINTER TEMPLATE %1 WITH INDEX=%2") | ||||
|                             .arg(fToWorkOn) | ||||
|                             .arg(templateIdx); | ||||
|                 } | ||||
|             } | ||||
|         } else if (fToWorkOn.contains("DC2C_cash", Qt::CaseInsensitive) | ||||
|                 && fToWorkOn.endsWith(".json", Qt::CaseInsensitive)) { | ||||
|             res = true; | ||||
|             if ((res = updateCashConf(fToWorkOn))) { | ||||
|                 qCritical() << QString("DOWNLOADED CASH TEMPLATE %1").arg(fToWorkOn); | ||||
|             } | ||||
|         } else if (fToWorkOn.contains("DC2C_conf", Qt::CaseInsensitive) | ||||
|                 && fToWorkOn.endsWith(".json", Qt::CaseInsensitive)) { | ||||
|             res = true; | ||||
|             if ((res= updateConfig(fToWorkOn))) { | ||||
|                 qCritical() << QString("DOWNLOADED CONFIG TEMPLATE %1").arg(fToWorkOn); | ||||
|             } | ||||
|         } else if (fToWorkOn.contains("DC2C_device", Qt::CaseInsensitive) | ||||
|                 && fToWorkOn.endsWith(".json", Qt::CaseInsensitive)) { | ||||
|             res = true; | ||||
|             if ((res = updateDeviceConf(fToWorkOn))) { | ||||
|                 qCritical() << QString("DOWNLOADED DEVICE TEMPLATE %1").arg(fToWorkOn); | ||||
|             } | ||||
|         } else { | ||||
|             qCritical() << "UNKNOWN JSON FILE NAME" << fToWorkOn; | ||||
|             res = false; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return res; | ||||
| } | ||||
|  | ||||
| bool Update::checkDownloadedJsonVersions(QStringList const& jsonFileNames) { | ||||
|     for (QStringList::size_type i=0; i < jsonFileNames.size(); ++i) { | ||||
|  | ||||
|         uint8_t jsonNr = 0; | ||||
|  | ||||
|         QString const &fName = jsonFileNames[i]; | ||||
|  | ||||
|         if (fName.endsWith("conf.json")) { | ||||
|             jsonNr = 1; | ||||
|         } else | ||||
|         if (fName.endsWith("device.json")) { | ||||
|             jsonNr = 2; | ||||
|         } else | ||||
|         if (fName.endsWith("cash.json")) { | ||||
|             jsonNr = 3; | ||||
|         } else { | ||||
|             QRegularExpressionMatch match; | ||||
|             static const QRegularExpression re("^(.*print)([0-3][0-9])\\.json\\s*$"); | ||||
|             int idx = fName.indexOf(re, 0, &match); | ||||
|             if (idx != -1) { | ||||
|                 QString captured = match.captured(match.lastCapturedIndex()); | ||||
|                 bool ok = false; | ||||
|                 int n = captured.toInt(&ok); | ||||
|                 if (ok) { | ||||
|                     jsonNr = n + 4; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (jsonNr != 0) { | ||||
| #if 0 | ||||
|             m_hw->sys_requestJsonVersions(jsonNr); | ||||
|             QThread::msleep(500); | ||||
|  | ||||
|             char buf[64]; | ||||
|             memset(buf, 0x00, sizeof(buf)); | ||||
|             m_hw->sys_getJsonVersions(jsonNr, buf); | ||||
|             buf[sizeof(buf)-1] = '\0'; | ||||
|  | ||||
|             QString const installedVersion(buf); | ||||
|             QString const fileVersion = getFileVersion(jsonFileNames[i]); | ||||
|  | ||||
|             qCritical() << "installed version:" << installedVersion; | ||||
|             qCritical() << "     file version:" << fileVersion; | ||||
|  | ||||
|             if (installedVersion == fileVersion) { | ||||
|  | ||||
|             } | ||||
| #endif | ||||
|  | ||||
|         } else { | ||||
|             qCritical() << "CANNOT FIND JSON-NR FOR" << fName; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| bool Update::downloadJson(enum FileTypeJson type, | ||||
|                           int templateIdx, | ||||
|                           QString jsFileToSendToDC) const { | ||||
|  | ||||
|     m_hw->dc_autoRequest(true); // downloading Json needs the AutoEmission flag | ||||
|     qDebug() << "SET AUTO-REQUEST=TRUE"; | ||||
|     QThread::sleep(1);  // make sure the auto-request flag is acknowledged | ||||
|  | ||||
|     QStringList lst; | ||||
|     bool ready = false; | ||||
|     int nTry = 25; | ||||
|     while ((ready = m_hw->sys_ready4sending()) == false) { | ||||
|         QThread::msleep(200); | ||||
|         if (--nTry <= 0) { | ||||
|             qCritical() << "SYS NOT READY FOR SENDING AFTER 5 SECONDS"; | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     bool ret = false; | ||||
|     QString msg; | ||||
|     lst.clear(); | ||||
|     if (ready) { | ||||
|         QFile file(jsFileToSendToDC); | ||||
|         QFileInfo fi(jsFileToSendToDC); // max. size of template file is 800 bytes | ||||
|         if (file.exists()) { | ||||
|             if (file.open(QIODevice::ReadOnly)) { | ||||
|                 if (fi.size() > 0 && fi.size() <= 800) { | ||||
|                     QByteArray ba = file.readAll(); | ||||
|                     // kindOfFile: 1=config, 2=device, 3=cash, 4=serial, 5=time, 6=printer | ||||
|                     //      nrOfTemplate=1...32 if kindOfFile==6 | ||||
|                     //      content = content of the Json file, max 800byte ascii signs | ||||
|                     if (m_hw->sys_sendJsonFileToDc((uint8_t)(type), | ||||
|                                                    templateIdx, | ||||
|                                                    (uint8_t *)ba.data())) { | ||||
|  | ||||
|                         /* | ||||
|                          *  Note: the machine id is contained in DC2C_conf.json. | ||||
|                          *  The idea was to use this to check if the download of | ||||
|                          *  the json-file was correct. It did not work, as the | ||||
|                          *  update of the PSA (to reflect a change in the | ||||
|                          *  machine id) did not happen immediately. | ||||
|                          * | ||||
|                             m_hw->dc_autoRequest(true); | ||||
|                             QThread::msleep(500); | ||||
|  | ||||
|                             // testing | ||||
|                             m_hw->request_ReadbackMachineID(); | ||||
|                             QThread::msleep(500); | ||||
|  | ||||
|                             uint8_t data[64]; | ||||
|                             memset(data, 0x00, sizeof(data)); | ||||
|                             uint8_t length = 0; | ||||
|  | ||||
|                             m_hw->readback_machineIDdata(&length, data); | ||||
|  | ||||
|                             QThread::msleep(500); | ||||
|  | ||||
|                             QByteArray ba((const char*)data, length); | ||||
|  | ||||
|                             qCritical() << length << "MACHINE ID =" << ba.toHex(':'); | ||||
|                         */ | ||||
|  | ||||
|                         ret = true; | ||||
|                     } else { | ||||
|                         qCritical() << QString("ERROR SEND JSON-FILE %1 TO DC").arg(file.fileName()); | ||||
|                     } | ||||
|                 } else { | ||||
|                     qCritical() << QString("SIZE OF %1 TOO BIG (%2 BYTES)").arg(jsFileToSendToDC).arg(fi.size()); | ||||
|                 } | ||||
|             } else { | ||||
|                 qCritical() << QString("CAN NOT OPEN ") + jsFileToSendToDC + " FOR READING"; | ||||
|             } | ||||
|         } else { | ||||
|             qCritical() << (QString(jsFileToSendToDC) + " DOES NOT EXIST"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     m_hw->dc_autoRequest(false); | ||||
|     qDebug() << "SET AUTO-REQUEST=FALSE"; | ||||
|     QThread::sleep(1);  // make sure the auto-request flag is acknowledged | ||||
|  | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| bool Update::updatePrinterTemplate(int templateIdx, QString jsFile) const { | ||||
|     return downloadJson(FileTypeJson::PRINTER, templateIdx, jsFile); | ||||
| } | ||||
|  | ||||
| bool Update::updateConfig(QString jsFile) { | ||||
|     return downloadJson(FileTypeJson::CONFIG, 0, jsFile); | ||||
| } | ||||
|  | ||||
| bool Update::updateCashConf(QString jsFile) { | ||||
|     return downloadJson(FileTypeJson::CASH, 0, jsFile); | ||||
| } | ||||
|  | ||||
| bool Update::updateDeviceConf(QString jsFile) { | ||||
|     return downloadJson(FileTypeJson::DEVICE, 0, jsFile); | ||||
| } | ||||
							
								
								
									
										107
									
								
								DownloadDCJsonFiles/update.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										107
									
								
								DownloadDCJsonFiles/update.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,107 @@ | ||||
| #ifndef UPDATE_H_INCLUDED | ||||
| #define UPDATE_H_INCLUDED | ||||
|  | ||||
| #include <QObject> | ||||
| #include <QString> | ||||
| #include <QFile> | ||||
| #include <QDir> | ||||
| #include <QByteArray> | ||||
| #include <QProcess> | ||||
| #include <QPluginLoader> | ||||
|  | ||||
| #include <DeviceController/interfaces.h> | ||||
|  | ||||
| #ifdef PTU5 | ||||
| #define SERIAL_PORT "ttymxc2" | ||||
| #else | ||||
| #define SERIAL_PORT "ttyUSB0" | ||||
| #endif | ||||
|  | ||||
| class Update : public QObject { | ||||
|     Q_OBJECT | ||||
|  | ||||
|     hwinf *m_hw = nullptr; | ||||
|     char const *m_serialInterface; | ||||
|     char const *m_baudrate; | ||||
|     QString m_customerRepository; | ||||
|     QString m_customerNrStr; | ||||
|     QString m_branchName; | ||||
|     QString m_pluginName; | ||||
|     QString m_workingDir; | ||||
|     bool m_maintenanceMode; | ||||
|     bool m_dryRun; | ||||
|     bool m_sys_areDCdataValid; | ||||
|  | ||||
|     static QPluginLoader pluginLoader; | ||||
|  | ||||
| public: | ||||
|     enum class DownloadResult {OK, ERROR, TIMEOUT, NOP}; | ||||
|     enum class FileTypeJson {CONFIG=1, DEVICE=2, CASH=3, SERIAL=4, TIME=5, PRINTER=6}; | ||||
|  | ||||
|     static hwinf *loadDCPlugin(QDir const &plugInDir, QString const &fn); | ||||
|     static bool unloadDCPlugin(); | ||||
|     static QStringList split(QString line, QChar sep = ','); | ||||
|  | ||||
|  | ||||
|     explicit Update(QString customerRepository, | ||||
|                     QString customerNrStr, | ||||
|                     QString branchName, | ||||
|                     QString plugInDir, | ||||
|                     QString pluginName, | ||||
|                     QString workingDir, | ||||
|                     bool dryRun = false, | ||||
|                     QObject *parent = nullptr, | ||||
|                     char const *serialInterface = SERIAL_PORT, | ||||
|                     char const *baudrate = "115200"); | ||||
|  | ||||
|     virtual ~Update() override; | ||||
|  | ||||
|     bool doUpdate(QStringList const &jsonFilesToDownload); | ||||
|  | ||||
|     bool updatePrinterTemplate(int templateIdx, QString fname) const; | ||||
|     bool updateConfig(QString jsFileToSendToDC); | ||||
|     bool updateCashConf(QString jsFileToSendToDC); | ||||
|     bool updateDeviceConf(QString jsFileToSendToDC); | ||||
|  | ||||
|     bool downloadJson(enum FileTypeJson type, int templateIdx, | ||||
|                       QString jsFileToSendToDC) const; | ||||
|  | ||||
|  | ||||
|     bool checkDownloadedJsonVersions(QStringList const& jsonFileNames); | ||||
| /* | ||||
|     bool checkDownloadedJsonVersions(QStringList const& jsonFileNames); | ||||
|  | ||||
|     hwinf *hw() { return m_hw; } | ||||
|     hwinf const *hw() const { return m_hw; } | ||||
|  | ||||
|     //QString customerId() { return m_customerId; } | ||||
|     //QString const customerId() const { return m_customerId; } | ||||
|  | ||||
|     QString branchName() { return m_branchName; } | ||||
|     QString const branchName() const { return m_branchName; } | ||||
|  | ||||
|     //QString repositoryPath() { return m_repositoryPath; } | ||||
|     //QString const repositoryPath() const { return m_repositoryPath; } | ||||
|  | ||||
| private: | ||||
|     static QString jsonType(enum FileTypeJson type); | ||||
|     bool openSerial(int br, QString baudrate, QString comPort) const; | ||||
|     void closeSerial() const; | ||||
|     bool isSerialOpen() const; | ||||
|     bool resetDeviceController() const; | ||||
|     QByteArray loadBinaryDCFile(QString filename) const; | ||||
|     bool downloadBinaryToDC(QString const &bFile) const; | ||||
|     bool updateBinary(QString const &fileToSendToDC); | ||||
|     QStringList getDcSoftAndHardWareVersion(); | ||||
|     QString getFileVersion(QString const& jsonFile); | ||||
|  | ||||
| private slots: | ||||
|     void readyReadStandardOutput(); | ||||
|     void readyReadStandardError(); | ||||
|     void finished(int exitCode, QProcess::ExitStatus exitStatus); | ||||
|     void onReportDCDownloadStatus(QString const &status); | ||||
|     void onReportDCDownloadSuccess(QString const &msg); | ||||
|     void onReportDCDownloadFailure(QString const &errorMsg); | ||||
| */ | ||||
| }; | ||||
| #endif // UPDATE_H_INCLUDED | ||||
| @@ -53,16 +53,59 @@ DEFINES += QT_DEPRECATED_WARNINGS | ||||
| #           local filesystem, json-files will be downloaded to firmware. | ||||
| #           The device-controller firmware will be handled in a later version. | ||||
| # 1.3.15:   Bug fixes found during testing. | ||||
| #           Don't disable Exit-button during update-process. | ||||
| #           Do not disable Exit-button during update-process. | ||||
| #           Removed worker-thread with an own event-loop: only the GUI thread | ||||
| #           has an event loop. Tested JSON-downloads several times successfully | ||||
| #           (using the slave lib where the CA helper tool was active as master). | ||||
| #           Turned previous worker-object into its own thread, but without any | ||||
| #           own event-loop (so it cannot block anything inside the CA-plugin). | ||||
| # 1.3.16:   Bug fixes found during testing. | ||||
| VERSION="1.3.17" | ||||
| #           Add ATBUpdateTool.ini and custom command line parser. | ||||
| 
 | ||||
| # 1.3.17:   Add ATBUpdateTool.ini and custom command line parser. Settings | ||||
| #           given in ATBUpdateTool.ini can be overwritten on the command-line. | ||||
| # 1.3.18:   Bug fixes found during testing. | ||||
| # 1.3.19:   Bug fixes found during testing. | ||||
| # 1.3.20:   Bug fixes found during testing. | ||||
| # 1.3.21:   Bug fixes found during testing: | ||||
| #           Fix directory of ATBUpdateTool.ini to be the working directory of | ||||
| #           the application rather than just ".". | ||||
| #           Check existance of etc-directory inside customer repository. | ||||
| #           Check for valid ISMAS trigger (button) 15x (=90s). | ||||
| #           NOTE: if the customer repository is cloned (or repaired and cloned | ||||
| #           again), and if the settings always-download-config=true and | ||||
| #           always-download-dc=true in the ATBUpdateTool.ini file, the download | ||||
| #           the printer-json files and the device controller file, even without | ||||
| #           an activated ISMAS trigger (button). The tariff-files are rsynced to | ||||
| #           the local filesystem for such clone. | ||||
| #           Set new filename for device controller: dc2c.bin. | ||||
| # 1.3.22:   Bug fixes found during testing: | ||||
| #           Fix the path-names of the json-files and the device-controller. | ||||
| #           Set automatic download of json-file in ATBUpdateTool.ini file for | ||||
| #           a fresh clone of the repository. | ||||
| # 1.3.23:   Added a 'break' to prevent a possible endless loop when checking if | ||||
| #           the device is alive. | ||||
| # | ||||
| # NOTE:     The versioning info has to be shifted up by one version, i.e. what | ||||
| #           happened for 1.3.23 was actually done in 1.3.24. | ||||
| # 1.3.24 | ||||
| # | ||||
| # 1.3._24_: Special version for szeged using a old dc-controller (4.42): | ||||
| #           Changes: | ||||
| #               (1) the ini-File now uses the libCAmaster.so. | ||||
| # 1.3._25_: Again special version for szeged, using interface.h, version 4.4. | ||||
| # 1.4.0   : Start with version at 1.4.0 (mainly to see a difference with Szeged) | ||||
| #           Set hash-value in EVENT-objects. Set location (project), version | ||||
| #           and info in send-last-version. | ||||
| #           If the customer repository does not exist, then do not check the | ||||
| #           ISMAS trigger, but proceed with the update procedure. Otherwise, | ||||
| #           check the ISMAS update-trigger as first step. | ||||
| #           If the current time is between 0.00 - 4.00 o'clock, then a wrong | ||||
| #           trigger-value will result in an UPDATE_STEP_NOT_NECESSARY. | ||||
| #           Move final processing to subclass UpdateProcessRunning. | ||||
| #           Disable EXIT-button for the whole update-process, except for the | ||||
| #           checking of the ISMAS-trigger-button (aka WAIT-button). | ||||
| # 1.4.1   : Sync files in the customer repository (under ./etc) as the very | ||||
| #           first step | ||||
| VERSION="1.4.1" | ||||
| # PLANNED TODOS: | ||||
| #   1:  Das Repository wird repariert bwz. neu geklont. Unabhaengig vom WAIT. | ||||
| #   2:  Wenn der WAIT-Button aktiv ist, dann wird ein Repository repariert (neu | ||||
| @@ -76,11 +119,48 @@ VERSION="1.3.17" | ||||
| #       ISMAS eine entsprechende Meldung anzeigen als Teil von SEND-LAST-VERSION. | ||||
| #       Wenn der WAIT-button aktiv ist, dann werden zumindest die opkg-commands | ||||
| #       ausgefuehrt. | ||||
| #   4:  rsync: immer alle Dateien soiegeln (bis auf opkg-commands) | ||||
| #   5:  Falls das Tool mal abstuerzt, dann einen Signal-Handler (fuer TERM) | ||||
| #       installieren, sodass zumnidest SEND-LAST-VERSION mit rausgeht. | ||||
| #   6:  rsync: explizites Binary, nicht das in busybox enthaltene. | ||||
| #   7:  Versionen der Json-Files lassen sich auslesen. | ||||
| #       Problem: Einstellungen in den Json-Files lassen sich auch mittels | ||||
| #       Funktionen in der CD-Library ueberschreiben. Damit ist dann wieder nicht | ||||
| #       mehr so klar, was jetzt eigentlich aktiv ist. | ||||
| #   8:  m_alwaysDownloadConfig und m_alwaysDownloadDC: vorbereitet: man koennte | ||||
| #       es so arrangieren, dass der DC plus die Json-files im Repository immer | ||||
| #       runtergeladen werden, obwohl sich im Repository gar nicts veraendert | ||||
| #       hat. Eeventuell nuetzlich beim initialen Setuo eines PSA. | ||||
| #   9:  Das Kunden-Repository sollte immer gezogen werden, unabhaengig von der | ||||
| #       Stellung des WAIT-Button. Grund: es koennte sein, dass andernfalls ein | ||||
| #       PSA weit hiter anderen steht, und dann ploetzlich einmal alle vorher- | ||||
| #       gehenden Aenderungen anzieht, die gar nicht fuer ihn gemeint waren. | ||||
| #  11:  Das Edit-Fenster teilen um die Anzeige zu verbessern. | ||||
| #  12:  Bei einem Update muss immer ersichtlich sein, warum es ueberhaupt | ||||
| #       angestossen wurde. Steht kein "WAIT" im ISMAS-Trigger, dann kann man | ||||
| #       davon ausgehen, dass es sich um ein automatisches Update handelt. | ||||
| #       In jedem Fall wird bei einem automatischen Update, bei dem der WAIT- | ||||
| #       Button nicht gesetzt war, ein "OK" gesendet, falls sonst nichts weiter | ||||
| #       zu tun ist. Beachte aber: wir haben auch noch den Fall, dass eine SD- | ||||
| #       Karte gesteckt wird. In diesem Fall wird ein komplettes Update gefahren, | ||||
| #       und zwar explizit auch ohne WAIT-Button. | ||||
| #       Am Ende eines Updates steht im ISMAS entweder ein "OK" oder ein "FAIL". | ||||
| # 13:   SendLastVersion: fuer jedes erfolgreich installierte Paket eine | ||||
| #       Send-Last-Version-Nachricht an ISMAS. Dadurch entsteht im ISMAS eine | ||||
| #       History (Christian darueber informieren). | ||||
| # 14:   Installiert werden nur Dateien, die neu sind oder geaendert wurden. | ||||
| #       Nicht etwas Dateien, die geloescht wurden: sicherstellen, dass man hier | ||||
| #       immer direkt im repository arbeitet, nicht auf dem Filesystem. | ||||
| #       Ferner: der DeviceController heisst dc2c.bin, auch fuer die Jsons | ||||
| #       sind Dtandard-Namen vergeben. Alternativ: alle vorhandenen Jsons | ||||
| #       werden runtergeladen: Thomas ist eh fuer deren Inhalte verantworlich. | ||||
| #       WICHTIG: immer ueberpruefen, ob die Dateien im Customer-Repository | ||||
| #       wirklich die richtigen Dateien sind. | ||||
| # 15:   Der WAIT-Button laesst sich auf WAIT zuruecksetzen (etwa wenn git | ||||
| #       selber Probleme hatte). | ||||
| # 16:   Der Download-Thread sollte sowohl die auto-Variable auf false setzen | ||||
| #       als auch den cycle-Timer stoppen, damit sichergestellt ist, dass der | ||||
| #       Download des DC nicht gestoert wird. | ||||
| 
 | ||||
| 
 | ||||
| win32 { | ||||
|     BUILD_DATE=$$system("date /t") | ||||
| @@ -107,7 +187,7 @@ DEFINES+=APP_EXTENDED_VERSION=\\\"$$EXTENDED_VERSION\\\" | ||||
| # keep comments, as /* fall through */ | ||||
| QMAKE_CXXFLAGS += -C | ||||
| QMAKE_CXXFLAGS += -g | ||||
| QMAKE_CXXFLAGS += -Wno-deprecated-copy | ||||
| QMAKE_CXXFLAGS += -Wno-deprecated-copy -O | ||||
| 
 | ||||
| contains( CONFIG, PTU5 ) { | ||||
|     greaterThan(QT_MAJOR_VERSION, 4): QT += serialport | ||||
| @@ -167,6 +247,7 @@ HEADERS += \ | ||||
|         process/command.h \ | ||||
|         message_handler.h \ | ||||
|         worker.h \ | ||||
|         interfaces.h \ | ||||
|         commandline_parser.h \ | ||||
|         plugins/interfaces.h | ||||
| 
 | ||||
							
								
								
									
										289
									
								
								UpdatePTUDevCtrl/UpdatePTUDevCtrl.pro
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										289
									
								
								UpdatePTUDevCtrl/UpdatePTUDevCtrl.pro
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,289 @@ | ||||
| QT += core gui | ||||
| QT += widgets serialport network | ||||
|  | ||||
| greaterThan(QT_MAJOR_VERSION, 4): QT += widgets | ||||
|  | ||||
| TARGET = ATBUpdateTool | ||||
|  | ||||
| # The following define makes your compiler emit warnings if you use | ||||
| # any Qt feature that has been marked deprecated (the exact warnings | ||||
| # depend on your compiler). Please consult the documentation of the | ||||
| # deprecated API in order to know how to port your code away from it. | ||||
| DEFINES += QT_DEPRECATED_WARNINGS | ||||
|  | ||||
| # You can also make your code fail to compile if it uses deprecated APIs. | ||||
| # In order to do so, uncomment the following line. | ||||
| # You can also select to disable deprecated APIs only up to a certain version of Qt. | ||||
| #DEFINES += QT_DISABLE_DEPRECATED_BEFORE=0x060000    # disables all the APIs deprecated before Qt 6.0.0 | ||||
|  | ||||
| # 1.3.6 :   Do not update device-controller/json files, but have the library | ||||
| #           (in a later step) do that. | ||||
| #           Fixed sending messages to ISMAS. | ||||
| #           Always execute contents of opkg_commands-file (even if there are no | ||||
| #           changes). | ||||
| # 1.3.7 :   Wait forever for git-commands to finish in QProcess executing such | ||||
| #           a command. | ||||
| # 1.3.8 :   Remove accessing opkg_commands under file-system-path /etc/psa_update. | ||||
| #           Activate download of json-configuration files. | ||||
| # 1.3.9 :   Fix sendLastVersion: use configured branch and not master branch in | ||||
| #               git show origin/master -s --format="c=%h m=%s d=%cI"    ==> | ||||
| #               git show origin/zg1/zone1 -s --format="c=%h m=%s d=%cI" | ||||
| #           Use dynamic values for os-release and apism-version when sending | ||||
| #           last version info. | ||||
| # 1.3.10:   Fix premature killing opkg-commands: detected timeout of 100s was | ||||
| #           too small when updating apism. | ||||
| #           Fix display of UPDATE_SUCCESS when opkg_command fails. Detected when | ||||
| #           updating apsim failed. | ||||
| # 1.3.11:   Integrate version of ATBUpdateTool in SendLastVersion-ISMAS-message. | ||||
| # 1.3.12:   Add command parameters for output of yocto-infos about ATBUpdateTool. | ||||
| #           Use 'git pull' instead of 'git fetch'. | ||||
| #           Use 'git clone --filter=blob:none' instead of 'git clone' to speed | ||||
| #           up cloning of customer repository. | ||||
| # 1.3.13:   Fix: if the customer repository is corrupted, remove it and re-clone | ||||
| #           the repository (without checking the ISMAS-trigger (WAIT-)button. | ||||
| # 1.3.14:   Add additional check for sanity of customer repository using | ||||
| #           "git fsck". | ||||
| #           Stream-lined code of update process: massive refactoring. | ||||
| #           Added functionality: If WAIT button is not active, then an existing | ||||
| #           customer repository will be repaired, or a not existing repository | ||||
| #           will be cloned. The process stops then. | ||||
| #           However, if the WAIT button is active, the at least the commands in | ||||
| #           opkg_commands will be executed. Changed files in the customer | ||||
| #           repository will be worked on: tariff-files will be synced with the | ||||
| #           local filesystem, json-files will be downloaded to firmware. | ||||
| #           The device-controller firmware will be handled in a later version. | ||||
| # 1.3.15:   Bug fixes found during testing. | ||||
| #           Do not disable Exit-button during update-process. | ||||
| #           Removed worker-thread with an own event-loop: only the GUI thread | ||||
| #           has an event loop. Tested JSON-downloads several times successfully | ||||
| #           (using the slave lib where the CA helper tool was active as master). | ||||
| #           Turned previous worker-object into its own thread, but without any | ||||
| #           own event-loop (so it cannot block anything inside the CA-plugin). | ||||
| # 1.3.16:   Bug fixes found during testing. | ||||
| # 1.3.17:   Add ATBUpdateTool.ini and custom command line parser. Settings | ||||
| #           given in ATBUpdateTool.ini can be overwritten on the command-line. | ||||
| # 1.3.18:   Bug fixes found during testing. | ||||
| # 1.3.19:   Bug fixes found during testing. | ||||
| # 1.3.20:   Bug fixes found during testing. | ||||
| # 1.3.21:   Bug fixes found during testing: | ||||
| #           Fix directory of ATBUpdateTool.ini to be the working directory of | ||||
| #           the application rather than just ".". | ||||
| #           Check existance of etc-directory inside customer repository. | ||||
| #           Check for valid ISMAS trigger (button) 15x (=90s). | ||||
| #           NOTE: if the customer repository is cloned (or repaired and cloned | ||||
| #           again), and if the settings always-download-config=true and | ||||
| #           always-download-dc=true in the ATBUpdateTool.ini file, the download | ||||
| #           the printer-json files and the device controller file, even without | ||||
| #           an activated ISMAS trigger (button). The tariff-files are rsynced to | ||||
| #           the local filesystem for such clone. | ||||
| #           Set new filename for device controller: dc2c.bin. | ||||
| # 1.3.22:   Bug fixes found during testing: | ||||
| #           Fix the path-names of the json-files and the device-controller. | ||||
| #           Set automatic download of json-file in ATBUpdateTool.ini file for | ||||
| #           a fresh clone of the repository. | ||||
| # 1.3.23:   Added a 'break' to prevent a possible endless loop when checking if | ||||
| #           the device is alive. | ||||
| # | ||||
| # NOTE:     The versioning info has to be shifted up by one version, i.e. what | ||||
| #           happened for 1.3.23 was actually done in 1.3.24. | ||||
| # 1.3.24 | ||||
| # | ||||
| # 1.3._24_: Special version for szeged using a old dc-controller (4.42): | ||||
| #           Changes: | ||||
| #               (1) the ini-File now uses the libCAmaster.so. | ||||
| # 1.3._25_: Again special version for szeged, using interface.h, version 4.4. | ||||
| # 1.4.0   : Start with version at 1.4.0 (mainly to see a difference with Szeged) | ||||
| #           Set hash-value in EVENT-objects. Set location (project), version | ||||
| #           and info in send-last-version. | ||||
| #           If the customer repository does not exist, then do not check the | ||||
| #           ISMAS trigger, but proceed with the update procedure. Otherwise, | ||||
| #           check the ISMAS update-trigger as first step. | ||||
| #           If the current time is between 0.00 - 4.00 o'clock, then a wrong | ||||
| #           trigger-value will result in an UPDATE_STEP_NOT_NECESSARY. | ||||
| #           Move final processing to subclass UpdateProcessRunning. | ||||
| #           Disable EXIT-button for the whole update-process, except for the | ||||
| #           checking of the ISMAS-trigger-button (aka WAIT-button). | ||||
| # 1.4.1   : Sync files in the customer repository (under ./etc) as the very | ||||
| #           first step | ||||
| # 1.4.2   : Do not check if <repo_dir>/etc/psa_tariff and /etc/psa_tariff are | ||||
| #           the same after an rsync. They might be noy after a change of the | ||||
| #           customer-number. | ||||
| # 1.4.3   : Use global directory for device-controller interfaces.h-file. | ||||
| # 1.4.4   : Add additional debug messages when downloading json-files. | ||||
| #           Move rsyncing of the customer-repository after the actual fetching | ||||
| #           of the repository. Otherwise, the update of, for instance | ||||
| #           tariff-files, would always be a step behind. | ||||
| # 1.4.5   : In case a new branch has been created in a remote | ||||
| #           customer-repository (e.g. origin/zg1/zone101), then fetch/pull | ||||
| #           this branch before switching to this now locally existen branch. | ||||
| #         : Improve output of GUI/Console and messages sent to ISMAS. | ||||
| #         : Fix: do not send the json-files to dc in any case. Bug introduced | ||||
| #           when pulling new branch, but branch already existed locally. | ||||
| # 1.4.6   : No exit()-call if loading CA-plugin fails. | ||||
| #           Check m_hw (pointer to CA-(Slave)Plugin) before its use. | ||||
| #           Only exit() if firmware is configured to be possibly updated, but | ||||
| #           loading the CA-plugin failed. | ||||
| VERSION="1.4.7" | ||||
| # PLANNED TODOS: | ||||
| #   1:  Das Repository wird repariert bwz. neu geklont. Unabhaengig vom WAIT. | ||||
| #   2:  Wenn der WAIT-Button aktiv ist, dann wird ein Repository repariert (neu | ||||
| #       geklont), aber zusaetzlich werden alle verfuegbaren Dateien als neu | ||||
| #       angesehen und die entsprechenden Aktionen durchgefuehrt: tariff-files | ||||
| #       spiegeln, json-files laden und dc laden. Also VORSICHT: das repository | ||||
| #       muss in diesem fall wirklich in ordnung sein. | ||||
| #   3:  Wurde keine Datei geaendert, kein initiales Clone und der WAIT-button | ||||
| #       nicht aktiv, so (passiert natuerlich nichts) kann man davon ausgehen, | ||||
| #       dass es sich um ein automatisches Update handelt. Dann koennte man im | ||||
| #       ISMAS eine entsprechende Meldung anzeigen als Teil von SEND-LAST-VERSION. | ||||
| #       Wenn der WAIT-button aktiv ist, dann werden zumindest die opkg-commands | ||||
| #       ausgefuehrt. | ||||
| #   5:  Falls das Tool mal abstuerzt, dann einen Signal-Handler (fuer TERM) | ||||
| #       installieren, sodass zumnidest SEND-LAST-VERSION mit rausgeht. | ||||
| #   6:  rsync: explizites Binary, nicht das in busybox enthaltene. | ||||
| #   7:  Versionen der Json-Files lassen sich auslesen. | ||||
| #       Problem: Einstellungen in den Json-Files lassen sich auch mittels | ||||
| #       Funktionen in der CD-Library ueberschreiben. Damit ist dann wieder nicht | ||||
| #       mehr so klar, was jetzt eigentlich aktiv ist. | ||||
| #   8:  m_alwaysDownloadConfig und m_alwaysDownloadDC: vorbereitet: man koennte | ||||
| #       es so arrangieren, dass der DC plus die Json-files im Repository immer | ||||
| #       runtergeladen werden, obwohl sich im Repository gar nicts veraendert | ||||
| #       hat. Eeventuell nuetzlich beim initialen Setuo eines PSA. | ||||
| #   9:  Das Kunden-Repository sollte immer gezogen werden, unabhaengig von der | ||||
| #       Stellung des WAIT-Button. Grund: es koennte sein, dass andernfalls ein | ||||
| #       PSA weit hiter anderen steht, und dann ploetzlich einmal alle vorher- | ||||
| #       gehenden Aenderungen anzieht, die gar nicht fuer ihn gemeint waren. | ||||
| #  11:  Das Edit-Fenster teilen um die Anzeige zu verbessern. | ||||
| #  12:  Bei einem Update muss immer ersichtlich sein, warum es ueberhaupt | ||||
| #       angestossen wurde. Steht kein "WAIT" im ISMAS-Trigger, dann kann man | ||||
| #       davon ausgehen, dass es sich um ein automatisches Update handelt. | ||||
| #       In jedem Fall wird bei einem automatischen Update, bei dem der WAIT- | ||||
| #       Button nicht gesetzt war, ein "OK" gesendet, falls sonst nichts weiter | ||||
| #       zu tun ist. Beachte aber: wir haben auch noch den Fall, dass eine SD- | ||||
| #       Karte gesteckt wird. In diesem Fall wird ein komplettes Update gefahren, | ||||
| #       und zwar explizit auch ohne WAIT-Button. | ||||
| #       Am Ende eines Updates steht im ISMAS entweder ein "OK" oder ein "FAIL". | ||||
| # 13:   SendLastVersion: fuer jedes erfolgreich installierte Paket eine | ||||
| #       Send-Last-Version-Nachricht an ISMAS. Dadurch entsteht im ISMAS eine | ||||
| #       History (Christian darueber informieren). | ||||
| # 14:   Installiert werden nur Dateien, die neu sind oder geaendert wurden. | ||||
| #       Nicht etwas Dateien, die geloescht wurden: sicherstellen, dass man hier | ||||
| #       immer direkt im repository arbeitet, nicht auf dem Filesystem. | ||||
| #       Ferner: der DeviceController heisst dc2c.bin, auch fuer die Jsons | ||||
| #       sind Dtandard-Namen vergeben. Alternativ: alle vorhandenen Jsons | ||||
| #       werden runtergeladen: Thomas ist eh fuer deren Inhalte verantworlich. | ||||
| #       WICHTIG: immer ueberpruefen, ob die Dateien im Customer-Repository | ||||
| #       wirklich die richtigen Dateien sind. | ||||
| # 15:   Der WAIT-Button laesst sich auf WAIT zuruecksetzen (etwa wenn git | ||||
| #       selber Probleme hatte). | ||||
| # 16:   Der Download-Thread sollte sowohl die auto-Variable auf false setzen | ||||
| #       als auch den cycle-Timer stoppen, damit sichergestellt ist, dass der | ||||
| #       Download des DC nicht gestoert wird. | ||||
|  | ||||
|  | ||||
| win32 { | ||||
|     BUILD_DATE=$$system("date /t") | ||||
|     BUILD_TIME=$$system("time /t") | ||||
| } else { | ||||
|     BUILD_DATE=$$system("date +%d-%m-%y") | ||||
|     BUILD_TIME=$$system("date +%H:%M:%S") | ||||
| } | ||||
|  | ||||
| GIT_COMMIT=$$system("git log -1 --format=oneline | cut -d' ' -f1") | ||||
|  | ||||
| EXTENDED_VERSION="$${VERSION}-$${GIT_COMMIT}" | ||||
|  | ||||
| INCLUDEPATH += plugins | ||||
|  | ||||
| CONFIG += c++17 | ||||
| # CONFIG -= app_bundle | ||||
|  | ||||
| DEFINES+=APP_VERSION=\\\"$$VERSION\\\" | ||||
| DEFINES+=APP_BUILD_DATE=\\\"$$BUILD_DATE\\\" | ||||
| DEFINES+=APP_BUILD_TIME=\\\"$$BUILD_TIME\\\" | ||||
| DEFINES+=APP_EXTENDED_VERSION=\\\"$$EXTENDED_VERSION\\\" | ||||
|  | ||||
| # keep comments, as /* fall through */ | ||||
| QMAKE_CXXFLAGS += -C | ||||
| QMAKE_CXXFLAGS += -g | ||||
| QMAKE_CXXFLAGS += -Wno-deprecated-copy -O | ||||
|  | ||||
| contains( CONFIG, PTU5 ) { | ||||
|     greaterThan(QT_MAJOR_VERSION, 4): QT += serialport | ||||
|     CONFIG += link_pkgconfig | ||||
|     lessThan(QT_MAJOR_VERSION, 5):   PKGCONFIG += qextserialport | ||||
|     QMAKE_CXXFLAGS += -O2 -std=c++17   # for GCC >= 4.7 | ||||
|  | ||||
|     PTU5BASEPATH = /opt/devel/ptu5 | ||||
|  | ||||
|     INCLUDEPATH += $$PTU5BASEPATH/qt/libs/devicecontroller/include | ||||
|     LIBS += -L$$PTU5BASEPATH/qt/libs/devicecontroller/library | ||||
|  | ||||
|     # QMAKE_CXXFLAGS += -Wno-deprecated-copy | ||||
|     ARCH = PTU5 | ||||
|     DEFINES+=PTU5 | ||||
| } | ||||
| contains( CONFIG, PTU5_YOCTO ) { | ||||
|     greaterThan(QT_MAJOR_VERSION, 4): QT += serialport | ||||
|     QMAKE_CXXFLAGS += -std=c++17   # for GCC >= 4.7 | ||||
|     # QMAKE_CXXFLAGS += -Wno-deprecated-copy | ||||
|     PTU5BASEPATH = /opt/devel/ptu5 | ||||
|     ARCH = PTU5 | ||||
|     DEFINES+=PTU5 | ||||
|  | ||||
|     LIBS += -lCAslave | ||||
|     LIBS += -lCAmaster | ||||
| } | ||||
| contains( CONFIG, DesktopLinux ) { | ||||
|     greaterThan(QT_MAJOR_VERSION, 4): QT += serialport | ||||
|     lessThan(QT_MAJOR_VERSION, 5):    CONFIG += extserialport | ||||
|     # QMAKE_CC = ccache $$QMAKE_CC | ||||
|     # QMAKE_CXX = ccache $$QMAKE_CXX | ||||
|     QMAKE_CXXFLAGS += -std=c++17 | ||||
|     # QMAKE_CXXFLAGS += -Wno-deprecated-copy | ||||
|     linux-clang {  QMAKE_CXXFLAGS += -Qunused-arguments   } | ||||
|     ARCH = DesktopLinux | ||||
|     DEFINES+=DesktopLinux | ||||
| } | ||||
|  | ||||
| SOURCES += \ | ||||
|         main.cpp \ | ||||
|         progress_event.cpp \ | ||||
|         update_dc_event.cpp \ | ||||
|         mainwindow.cpp \ | ||||
|         utils.cpp \ | ||||
|         update.cpp \ | ||||
|         git/git_client.cpp \ | ||||
|         ismas/ismas_client.cpp \ | ||||
|         process/command.cpp \ | ||||
|         message_handler.cpp \ | ||||
|         worker.cpp \ | ||||
|         commandline_parser.cpp | ||||
|  | ||||
| HEADERS += \ | ||||
|         update.h \ | ||||
|         progress_event.h \ | ||||
|         update_dc_event.h \ | ||||
|         utils.h \ | ||||
|         mainwindow.h \ | ||||
|         git/git_client.h \ | ||||
|         apism/ismas_data.h \ | ||||
|         ismas/ismas_client.h \ | ||||
|         process/command.h \ | ||||
|         message_handler.h \ | ||||
|         worker.h \ | ||||
|         commandline_parser.h | ||||
|  | ||||
| OTHER_FILES += \ | ||||
|     ATBUpdateTool.ini | ||||
|  | ||||
| FORMS += \ | ||||
|     mainwindow.ui | ||||
|  | ||||
|  | ||||
| ########################################################################################## | ||||
| # for running program on target through QtCreator | ||||
| contains( CONFIG, PTU5 ) { | ||||
|    qnx: target.path = /tmp/$${TARGET}/bin | ||||
|    else: unix:!android: target.path = /opt/app/tools/atbupdate/ | ||||
|    !isEmpty(target.path): INSTALLS += target | ||||
| } | ||||
| @@ -6,16 +6,18 @@ | ||||
| #include <QFile> | ||||
| 
 | ||||
| CommandLineParser::CommandLineParser() | ||||
|   : m_repositoryUrl("") | ||||
|   , m_plugInDir("") | ||||
|   , m_plugInName("") | ||||
|   , m_workingDir("") | ||||
|   , m_dryRun("") | ||||
|   , m_noUpdatePsaHardware("") | ||||
|   , m_showYoctoVersion("") | ||||
|   , m_showYoctoInstallStatus("") | ||||
|   , m_showExtendedVersion("") | ||||
|   , m_iniFileName("") | ||||
|   : m_repositoryUrl("https://git.mimbach49.de/GerhardHoffmann") | ||||
|   , m_plugInDir("/usr/lib/") | ||||
|   , m_plugInName("libCAslave.so") | ||||
|   , m_workingDir("/opt/app/tools/atbupdate/") | ||||
|   , m_dryRun("false") | ||||
|   , m_noUpdatePsaHardware("false") | ||||
|   , m_showYoctoVersion("false") | ||||
|   , m_showYoctoInstallStatus("false") | ||||
|   , m_showExtendedVersion("false") | ||||
|   , m_iniFileName("ATBUpdateTool.ini") | ||||
|   , m_alwaysDownloadConfig("false") | ||||
|   , m_alwaysDownloadDC("false") | ||||
|   , m_repositoryUrlOption( | ||||
|         QCommandLineOption( | ||||
|             QStringList() << "repository-url" << "repository-url", | ||||
| @@ -45,6 +47,14 @@ CommandLineParser::CommandLineParser() | ||||
|         QCommandLineOption( | ||||
|             "no-psa-hardware-update", | ||||
|             QCoreApplication::translate("main", "Do not update the PSA firmware (json, device-controller)."))) | ||||
|   , m_alwaysDownloadConfigOption( | ||||
|         QCommandLineOption( | ||||
|             "always-download-config", | ||||
|             QCoreApplication::translate("main", "Always download the (json-)configs to DC)."))) | ||||
|   , m_alwaysDownloadDCOption( | ||||
|         QCommandLineOption( | ||||
|             "always-download-dc", | ||||
|             QCoreApplication::translate("main", "Always download the dc-bin-file to DC)."))) | ||||
|   , m_workingDirectoryOption( | ||||
|         QCommandLineOption( | ||||
|             QStringList() << "working-directory" << "working-directory", | ||||
| @@ -77,26 +87,40 @@ void CommandLineParser::configure() { | ||||
|     m_repositoryUrlOption.setDefaultValue("https://git.mimbach49.de/GerhardHoffmann"); | ||||
|     m_parser.addOption(m_repositoryUrlOption); | ||||
| 
 | ||||
|     m_iniFileDirectoryOption.setDefaultValue("."); | ||||
|     m_iniFileDirectoryOption.setDefaultValue(QCoreApplication::applicationDirPath()); | ||||
|     m_parser.addOption(m_iniFileDirectoryOption); | ||||
| 
 | ||||
|     m_iniFileNameOption.setDefaultValue("ATBUpdateTool.ini"); | ||||
|     m_parser.addOption(m_iniFileNameOption); | ||||
| 
 | ||||
|     m_pluginDirectoryOption.setDefaultValue("/usr/lib"); | ||||
|     m_pluginDirectoryOption.setDefaultValue("/usr/lib/"); | ||||
|     m_parser.addOption(m_pluginDirectoryOption); | ||||
| 
 | ||||
|     m_pluginNameOption.setDefaultValue("libCAslave.so"); | ||||
|     m_parser.addOption(m_pluginNameOption); | ||||
| 
 | ||||
|     m_alwaysDownloadConfigOption.setDefaultValue("false"); | ||||
|     m_parser.addOption(m_alwaysDownloadConfigOption); | ||||
| 
 | ||||
|     m_alwaysDownloadDCOption.setDefaultValue("false"); | ||||
|     m_parser.addOption(m_alwaysDownloadDCOption); | ||||
| 
 | ||||
|     m_noDownloadOption.setDefaultValue("false"); | ||||
|     m_parser.addOption(m_noDownloadOption); | ||||
| 
 | ||||
|     m_workingDirectoryOption.setDefaultValue("/opt/app/tools/atbupdate"); | ||||
|     m_workingDirectoryOption.setDefaultValue("/opt/app/tools/atbupdate/"); | ||||
|     m_parser.addOption(m_workingDirectoryOption); | ||||
| 
 | ||||
|     m_dryRunOption.setDefaultValue("false"); | ||||
|     m_parser.addOption(m_dryRunOption); | ||||
| 
 | ||||
|     m_extendedVersionOption.setDefaultValue("false"); | ||||
|     m_parser.addOption(m_extendedVersionOption); | ||||
| 
 | ||||
|     m_yoctoVersionOption.setDefaultValue("false"); | ||||
|     m_parser.addOption(m_yoctoVersionOption); | ||||
| 
 | ||||
|     m_yoctoInstallStatusOption.setDefaultValue("false"); | ||||
|     m_parser.addOption(m_yoctoInstallStatusOption); | ||||
| } | ||||
| 
 | ||||
| @@ -105,12 +129,19 @@ void CommandLineParser::readSettings() { | ||||
|     QString const iniFileName = m_parser.value(m_iniFileNameOption); | ||||
|     m_iniFileName = QDir::cleanPath(iniFileDir + QDir::separator() + iniFileName); | ||||
| 
 | ||||
|     qCritical() << __PRETTY_FUNCTION__ << " iniFileDir" << iniFileDir; | ||||
|     qCritical() << __PRETTY_FUNCTION__ << "iniFileName" << m_iniFileName; | ||||
| 
 | ||||
|     if (!m_iniFileName.isEmpty()) { | ||||
|         if (QFile(m_iniFileName).exists()) { | ||||
|             QSettings settings(m_iniFileName, QSettings::IniFormat); | ||||
|             QStringList keys = settings.allKeys(); | ||||
|             for (QString const &key: keys) { | ||||
|                 QVariant v = settings.value(key); | ||||
| 
 | ||||
|                 qCritical() << __PRETTY_FUNCTION__ | ||||
|                             << key << " -> " << v.toString(); | ||||
| 
 | ||||
|                 if (key.contains("repository-url")) { | ||||
|                     m_repositoryUrl = v.toString(); | ||||
|                 } else | ||||
| @@ -118,7 +149,7 @@ void CommandLineParser::readSettings() { | ||||
|                     m_plugInDir = v.toString(); | ||||
|                 } else | ||||
|                 if (key.contains("working-directory")) { | ||||
|                     m_plugInName = v.toString(); | ||||
|                     m_workingDir = v.toString(); | ||||
|                 } else | ||||
|                 if (key.contains("dry-run")) { | ||||
|                     m_dryRun = (v.toBool() ? "true" : "false"); | ||||
| @@ -129,6 +160,12 @@ void CommandLineParser::readSettings() { | ||||
|                 if (key.contains("no-psa-hardware-update")) { | ||||
|                     m_noUpdatePsaHardware = (v.toBool() ? "true" : "false"); | ||||
|                 } else | ||||
|                 if (key.contains("always-download-config")) { | ||||
|                     m_alwaysDownloadConfig = (v.toBool() ? "true" : "false"); | ||||
|                 } else | ||||
|                 if (key.contains("always-download-dc")) { | ||||
|                     m_alwaysDownloadDC = (v.toBool() ? "true" : "false"); | ||||
|                 } else | ||||
|                 if (key.contains("yocto-install")) { | ||||
|                     m_showYoctoInstallStatus = (v.toBool() ? "true" : "false"); | ||||
|                 } else | ||||
| @@ -137,72 +174,93 @@ void CommandLineParser::readSettings() { | ||||
|                 } else | ||||
|                 if (key.contains("plugin-name")) { | ||||
|                     m_plugInName = v.toString(); | ||||
|                 } else { | ||||
|                     qCritical() << __PRETTY_FUNCTION__ | ||||
|                                 << key << " -> (UNKNOWN) " << v.toString(); | ||||
|                 } | ||||
|             } | ||||
|         } else { | ||||
|             qCritical() << __PRETTY_FUNCTION__ << "iniFileName" << m_iniFileName | ||||
|                         << "DOES NOT EXIST"; | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| QString CommandLineParser::repositoryUrl() { | ||||
|     return m_repositoryUrl; | ||||
|     if (m_repositoryUrl.isEmpty()) { | ||||
|     if (m_parser.isSet(m_repositoryUrlOption)) { | ||||
|         m_repositoryUrl = m_parser.value(m_repositoryUrlOption); | ||||
|     } | ||||
|     return m_repositoryUrl; | ||||
| } | ||||
| 
 | ||||
| QString CommandLineParser::plugInDir() { | ||||
|     if (m_plugInDir.isEmpty()) { | ||||
|     if (m_parser.isSet(m_pluginDirectoryOption)) { | ||||
|         m_plugInDir = m_parser.value(m_pluginDirectoryOption); | ||||
|     } | ||||
|     return m_plugInDir; | ||||
| } | ||||
| 
 | ||||
| QString CommandLineParser::plugInName() { | ||||
|     if (m_plugInName.isEmpty()) { | ||||
|     if (m_parser.isSet(m_pluginNameOption)) { | ||||
|         m_plugInName = m_parser.value(m_pluginNameOption); | ||||
|     } | ||||
|     return m_plugInName; | ||||
| } | ||||
| 
 | ||||
| QString CommandLineParser::workingDir() { | ||||
|     if (m_workingDir.isEmpty()) { | ||||
|     if (m_parser.isSet(m_workingDirectoryOption)) { | ||||
|         m_workingDir = m_parser.value(m_workingDirectoryOption); | ||||
|     } | ||||
|     return m_workingDir; | ||||
| } | ||||
| 
 | ||||
| bool CommandLineParser::dryRun() { | ||||
|     if (m_dryRun.isEmpty()) { | ||||
|     if (m_parser.isSet(m_dryRunOption)) { | ||||
|         m_dryRun = m_parser.value(m_dryRunOption); | ||||
|     } | ||||
|     return m_dryRun == "false" ? false : true; | ||||
| } | ||||
| 
 | ||||
| bool CommandLineParser::noUpdatePsaHardware() { | ||||
|     if (m_noUpdatePsaHardware.isEmpty()) { | ||||
|     if (m_parser.isSet(m_noDownloadOption)) { | ||||
|         m_noUpdatePsaHardware = m_parser.value(m_noDownloadOption); | ||||
|     } | ||||
|     return m_noUpdatePsaHardware == "false" ? false : true; | ||||
| } | ||||
| 
 | ||||
| bool CommandLineParser::yoctoVersion() { | ||||
|     if (m_showYoctoVersion.isEmpty()) { | ||||
|     if (m_parser.isSet(m_yoctoVersionOption)) { | ||||
|         m_showYoctoVersion = m_parser.value(m_yoctoVersionOption); | ||||
|     } | ||||
|     return m_showYoctoVersion == "false" ? false : true; | ||||
| } | ||||
| 
 | ||||
| bool CommandLineParser::yoctoInstallStatus() { | ||||
|     if (m_showYoctoInstallStatus.isEmpty()) { | ||||
|     if (m_parser.isSet(m_yoctoInstallStatusOption)) { | ||||
|         m_showYoctoInstallStatus = m_parser.value(m_yoctoInstallStatusOption); | ||||
|     } | ||||
|     return m_showYoctoInstallStatus == "false" ? false : true; | ||||
| } | ||||
| 
 | ||||
| bool CommandLineParser::extendedVersion() { | ||||
|     if (m_showExtendedVersion.isEmpty()) { | ||||
|     if (m_parser.isSet(m_extendedVersionOption)) { | ||||
|         m_showExtendedVersion = m_parser.value(m_extendedVersionOption); | ||||
|     } | ||||
|     return m_showExtendedVersion == "false" ? false : true; | ||||
| } | ||||
| 
 | ||||
| bool CommandLineParser::alwaysDownloadConfig() { | ||||
|     if (m_parser.isSet(m_alwaysDownloadConfigOption)) { | ||||
|         m_alwaysDownloadConfig = m_parser.value(m_alwaysDownloadConfigOption); | ||||
|         qCritical() << "m_alwaysDownloadConfigOption IS SET" << m_alwaysDownloadConfig; | ||||
|     } | ||||
|     qCritical() << "m_alwaysDownloadConfig" << m_alwaysDownloadConfig; | ||||
|     return m_alwaysDownloadConfig == "false" ? false : true; | ||||
| } | ||||
| 
 | ||||
| bool CommandLineParser::alwaysDownloadDC() { | ||||
|     if (m_parser.isSet(m_alwaysDownloadDCOption)) { | ||||
|         m_alwaysDownloadDC = m_parser.value(m_alwaysDownloadDCOption); | ||||
|     } | ||||
|     return m_alwaysDownloadDC == "false" ? false : true; | ||||
| } | ||||
| @@ -17,6 +17,8 @@ class CommandLineParser : public QCommandLineParser { | ||||
|     QString m_showYoctoInstallStatus; | ||||
|     QString m_showExtendedVersion; | ||||
|     QString m_iniFileName; | ||||
|     QString m_alwaysDownloadConfig; | ||||
|     QString m_alwaysDownloadDC; | ||||
| 
 | ||||
|     QCommandLineOption m_repositoryUrlOption; | ||||
|     QCommandLineOption m_iniFileDirectoryOption; | ||||
| @@ -24,6 +26,8 @@ class CommandLineParser : public QCommandLineParser { | ||||
|     QCommandLineOption m_pluginDirectoryOption; | ||||
|     QCommandLineOption m_pluginNameOption; | ||||
|     QCommandLineOption m_noDownloadOption; | ||||
|     QCommandLineOption m_alwaysDownloadConfigOption; | ||||
|     QCommandLineOption m_alwaysDownloadDCOption; | ||||
|     QCommandLineOption m_workingDirectoryOption; | ||||
|     QCommandLineOption m_dryRunOption; | ||||
|     QCommandLineOption m_extendedVersionOption; | ||||
| @@ -53,5 +57,7 @@ public: | ||||
|     bool yoctoVersion(); | ||||
|     bool yoctoInstallStatus(); | ||||
|     bool extendedVersion(); | ||||
|     bool alwaysDownloadConfig(); | ||||
|     bool alwaysDownloadDC(); | ||||
| }; | ||||
| #endif // COMMAND_LINE_PARSER_H_INCLUDED
 | ||||
							
								
								
									
										
											BIN
										
									
								
								UpdatePTUDevCtrl/doc/ATBUpdateTool.drawio-4.pdf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								UpdatePTUDevCtrl/doc/ATBUpdateTool.drawio-4.pdf
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								UpdatePTUDevCtrl/doc/ATBUpdateTool4.pdf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								UpdatePTUDevCtrl/doc/ATBUpdateTool4.pdf
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -6,6 +6,7 @@ | ||||
| #include <QRegularExpression> | ||||
| #include <QDebug> | ||||
| #include <QDir> | ||||
| #include <QStringList> | ||||
| 
 | ||||
| 
 | ||||
| GitClient::GitClient(QString const &customerNrStr, | ||||
| @@ -192,7 +193,7 @@ QStringList GitClient::gitShowReason(QString branchName) { | ||||
|             int const m = s.indexOf("m="); | ||||
|             int const d = s.indexOf("d="); | ||||
| 
 | ||||
|             QString msg = IsmasClient::getReasonForLastSendVersion(); | ||||
|             QString msg{""}; // IsmasClient::getReasonForLastSendVersion();
 | ||||
|             QString commit{""}, date{""}; | ||||
| 
 | ||||
|             if (c != -1) { | ||||
| @@ -204,9 +205,9 @@ QStringList GitClient::gitShowReason(QString branchName) { | ||||
|                     start = m + 2; | ||||
|                     if (d >= start) { | ||||
|                         length = d - start; | ||||
|                         msg += " ("; | ||||
|                         // msg = " (";
 | ||||
|                         msg = s.mid(start, length).trimmed(); | ||||
|                         msg += ")"; | ||||
|                         // msg += ")";
 | ||||
| 
 | ||||
|                         start = d + 2; | ||||
|                         date = s.mid(start); | ||||
| @@ -290,6 +291,100 @@ bool GitClient::gitFsck() { | ||||
|     } | ||||
|     return r; | ||||
| } | ||||
| 
 | ||||
| bool GitClient::branchExistsRemotely() { | ||||
|     bool remoteBranchExists = false; | ||||
|     if (QDir(m_customerRepository).exists()) { | ||||
|         qInfo() << "BRANCH NAME" << m_branchName; | ||||
|         QString const cmd = QString("git ls-remote --exit-code --heads origin %1").arg(m_branchName); | ||||
|         Command c(cmd); | ||||
|         if (c.execute(m_customerRepository)) { | ||||
|             // expected result: c16c833c8778c1b3691a74afee5a469177e4e69b refs/heads/zg1/zone1000
 | ||||
|             QString const s = c.getCommandResult().trimmed(); | ||||
| 
 | ||||
| 
 | ||||
|             if (!s.isEmpty()) { | ||||
|                 // the result is only one line
 | ||||
|                 if ((remoteBranchExists = s.contains(m_branchName)) == true) { | ||||
|                     qCritical() << "(" << __func__ << ":" << __LINE__ << ") branch" | ||||
|                                 << m_branchName << "EXISTS REMOTELY. (" << s << ")"; | ||||
|                 } | ||||
|             } else { | ||||
|                 Utils::printCriticalErrorMsg(QString("EMPTY RESULT FOR CMD %1").arg(cmd)); | ||||
|             } | ||||
|         } else { | ||||
|             Utils::printCriticalErrorMsg(QString("FAILED TO EXEC '%1'").arg(cmd)); | ||||
|         } | ||||
|     } | ||||
|     return remoteBranchExists; | ||||
| } | ||||
| 
 | ||||
| bool GitClient::branchExistsLocally() { | ||||
|     Command c("git branch -l"); | ||||
|     if (c.execute(m_customerRepository)) { | ||||
|         QString const s = c.getCommandResult().trimmed(); | ||||
|         if (!s.isEmpty()) { | ||||
|             QStringList lines = Update::split(s, '\n'); | ||||
|             if (!lines.empty()) { | ||||
|                 for (int i=0; i < lines.size(); ++i) { | ||||
|                     QString line = lines.at(i); | ||||
|                     // expected: * [new branch]      zg1/zone12 -> origin/zg1/zone12"
 | ||||
|                     if (line.contains(m_branchName)) { | ||||
|                         if (m_worker) { | ||||
|                             QStringList lst(QString("BRANCH-NAME %1 CONTAINED IN RESULT %2").arg(m_branchName).arg(s)); | ||||
|                             m_worker->CONSOLE(lst) << Worker::UPDATE_STEP::PULL_NEW_BRANCH; | ||||
|                         } | ||||
|                         return true; | ||||
|                     } | ||||
|                 } | ||||
| 
 | ||||
|                 if (m_worker) { | ||||
|                     QStringList lst(QString("BRANCH-NAME %1 NOT CONTAINED IN RESULT %2").arg(m_branchName).arg(s)); | ||||
|                     m_worker->CONSOLE(lst) << Worker::UPDATE_STEP::PULL_NEW_BRANCH_FAILURE; | ||||
|                 } | ||||
| 
 | ||||
|             } else { | ||||
|                 if (m_worker) { | ||||
|                     QStringList lst(QString("'git branch -l' RETURNED NO LINES")); | ||||
|                     m_worker->CONSOLE(lst) << Worker::UPDATE_STEP::PULL_NEW_BRANCH_FAILURE; | ||||
|                 } | ||||
|             } | ||||
|         } else { | ||||
|             if (m_worker) { | ||||
|                 QStringList lst(QString("'git branch -l' RETURNED EMPTY RESULT")); | ||||
|                 m_worker->CONSOLE(lst) << Worker::UPDATE_STEP::PULL_NEW_BRANCH_FAILURE; | ||||
|             } | ||||
|         } | ||||
|     } else { | ||||
|         if (m_worker) { | ||||
|             QStringList lst(QString("FAILED TO EXEC 'git branch -l'")); | ||||
|             m_worker->CONSOLE(lst) << Worker::UPDATE_STEP::PULL_NEW_BRANCH_FAILURE; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     return false; | ||||
| } | ||||
| 
 | ||||
| bool GitClient::gitPullNewBranches() { | ||||
| 
 | ||||
|     if (QDir(m_customerRepository).exists()) { | ||||
|         Command c("git pull"); | ||||
|         if (c.execute(m_customerRepository)) { | ||||
|             QString const s = c.getCommandResult().trimmed(); | ||||
|             // expected: Already up-to-date.
 | ||||
|             if (!s.isEmpty()) { | ||||
|                 QStringList lst; | ||||
|                 QString msg(QString("GIT-PULL-NEW-BRANCH. RESULT=%1").arg(s)); | ||||
|                 if (m_worker) { | ||||
|                     m_worker->CONSOLE(lst) << Worker::UPDATE_STEP::PULL_NEW_BRANCH; | ||||
|                 } | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     return false; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  Hat sich nichts geaendert, so werden auch keine Commits <>..<> angezeigt | ||||
|  */ | ||||
| @@ -316,6 +411,31 @@ std::optional<QString> GitClient::gitPull() { | ||||
|                         //   6ed893f..5d9882c  zg1/zone2  -> origin/zg1/zone2
 | ||||
|                         //   4384d17..77045d8  zg1/zone3  -> origin/zg1/zone3
 | ||||
|                         //   89d2812..36a0d74  zg1/zone5  -> origin/zg1/zone5
 | ||||
|                         //
 | ||||
|                         // More exactly:
 | ||||
|                         // remote: Counting objects: 382, done.
 | ||||
|                         // remote: Compressing objects: 100% (203/203), done.
 | ||||
|                         // remote: Total 278 (delta 177), reused 103 (delta 59)
 | ||||
|                         // Receiving objects: 100% (278/278), 4.89 MiB | 539 KiB/s, done.
 | ||||
|                         // Resolving deltas: 100% (177/177), completed with 40 local objects.
 | ||||
|                         // From ssh://longair@pacific.mpi-cbg.de/srv/git/fiji
 | ||||
|                         // 3036acc..9eb5e40  debian-release-20081030 -> origin/debian-release-20081030
 | ||||
|                         // * [new branch]      debian-release-20081112 -> origin/debian-release-20081112
 | ||||
|                         // * [new branch]      debian-release-20081112.1 -> origin/debian-release-20081112.1
 | ||||
|                         // 3d619e7..6260626  master     -> origin/master
 | ||||
|                         //
 | ||||
|                         // The most important bits here are the lines like these:
 | ||||
|                         //
 | ||||
|                         //     3036acc..9eb5e40  debian-release-20081030 -> origin/debian-release-20081030
 | ||||
|                         //   * [new branch]      debian-release-20081112 -> origin/debian-release-20081112
 | ||||
|                         //
 | ||||
|                         // The first line of these two shows that your remote-tracking branch
 | ||||
|                         // origin/debian-release-20081030 has been advanced from the commit 3036acc to 9eb5e40.
 | ||||
|                         // The bit before the arrow is the name of the branch in the remote repository.
 | ||||
|                         // The second line similarly show that since we last did this, a new remote tracking
 | ||||
|                         // branch has been created. (git fetch may also fetch new tags if they have appeared
 | ||||
|                         // in the remote repository.)
 | ||||
| 
 | ||||
|                         bool found = false; | ||||
|                         for (int i=0; i < lines.size(); ++i) { | ||||
|                             if (lines.at(i).contains(m_branchName)) { | ||||
| @@ -57,6 +57,10 @@ class GitClient : public QObject { | ||||
|     static QString gitBlob(QString fileName); | ||||
|     QString gitCommitForBlob(QString blob); | ||||
|     bool gitIsFileTracked(QString file2name); | ||||
| 
 | ||||
|     bool branchExistsRemotely(); | ||||
|     bool branchExistsLocally(); | ||||
|     bool gitPullNewBranches(); | ||||
| }; | ||||
| 
 | ||||
| #endif // GIT_CLIENT_H_INCLUDED
 | ||||
| @@ -384,7 +384,7 @@ QString IsmasClient::errorBackendNotConnected(QString const &info, | ||||
|                                               QString const &version) { | ||||
|     return updateNewsToIsmas("U0003", | ||||
|                              m_progressInPercent, | ||||
|                              RESULT_CODE::INSTALL_ERROR, | ||||
|                              RESULT_CODE::ISMAS_NO_CONNECTION_ERROR, | ||||
|                              "CHECK BACKEND CONNECTIVITY", | ||||
|                              info.toStdString().c_str(), | ||||
|                              version.toStdString().c_str()); | ||||
| @@ -394,7 +394,7 @@ QString IsmasClient::errorGitClone(QString const &info, | ||||
|                                    QString const &version) { | ||||
|     return updateNewsToIsmas("U0003", | ||||
|                              m_progressInPercent, | ||||
|                              RESULT_CODE::INSTALL_ERROR, | ||||
|                              RESULT_CODE::GIT_CLONE_ERROR, | ||||
|                              "CLONE CUSTOMER REPOSITORY FAILED", | ||||
|                              info.toStdString().c_str(), | ||||
|                              version.toStdString().c_str()); | ||||
| @@ -442,7 +442,7 @@ QString IsmasClient::updateTriggerSet(QString const &info, QString const &versio | ||||
| QString IsmasClient::errorUpdateTrigger(QString const &info, QString const &version) { | ||||
|     return updateNewsToIsmas("U0003", | ||||
|                              m_progressInPercent, | ||||
|                              RESULT_CODE::INSTALL_ERROR, | ||||
|                              RESULT_CODE::ISMAS_TRIGGER_ERROR, | ||||
|                              "CHECK UPDATE TRIGGER", | ||||
|                              info.toStdString().c_str(), | ||||
|                              version.toStdString().c_str()); | ||||
| @@ -793,6 +793,152 @@ QString IsmasClient::updateOfPSASendVersion(PSAInstalled const &psa) { | ||||
|     return buf; | ||||
| } | ||||
| 
 | ||||
| #if 0 | ||||
| // prepare
 | ||||
| QString IsmasClient::sendLastVersion(UPDATE_COMPONENT updateComponent, | ||||
|                                      PSAInstalled const &psa) { | ||||
|     static char buf[4096*2]; | ||||
|     memset(buf, 0, sizeof(buf)); | ||||
| 
 | ||||
|     switch (updateComponent) { | ||||
|     case UPDATE_COMPONENT::TARIFF: | ||||
|     snprintf(buf, sizeof(buf)-1, | ||||
|         "{" | ||||
|             "\"VERSION_INFO\" : {" | ||||
|                 "\"UPDATE_REASON\":\"%s\"," | ||||
|                 "\"CREATED\":\"%s\"," | ||||
|                 "\"GIT_COMMIT\":\"%s\"" | ||||
|             "}," | ||||
|             "\"TARIFF\" : {" | ||||
|                 "\"VERSION\" : \"%s\"," | ||||
|                 "\"PROJECT\" : \"%s\"," | ||||
|                 "\"ZONE\" : %d," | ||||
|                 "\"INFO\" : \"%s\"," | ||||
|                 "\"SIZE\" : %d\"" | ||||
|             "}," | ||||
|         "}", | ||||
|         psa.tariff.versionInfo.reason.toStdString().c_str(), | ||||
|         psa.tariff.versionInfo.created.toStdString().c_str(), | ||||
|         psa.tariff.versionInfo.lastCommit.toStdString().c_str(), | ||||
| 
 | ||||
|         psa.tariff.version.toStdString().c_str(), | ||||
|         psa.tariff.project.toStdString().c_str(), | ||||
|         psa.tariff.zone, | ||||
|         psa.tariff.info.toStdString().c_str(), | ||||
|         psa.tariff.size); | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::SOFTWARE_ATBQT: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::SOFTWARE_APISM: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::SOFTWARE_ATB_UPDATE_TOOL: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::CONFIG_PTU5_CPU_SERIAL: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::CONFIG_DEVICE_CONTROLLER: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::CONFIG_PRINTER: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::CONFIG_BNA: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::PLUGIN_ATB_DEVICE_CONTROLLER: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::PLUGIN_INGENICO_CC: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::PLUGIN_MOBILISIS_CALC_PRICE: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::PLUGIN_MOBILISIS_CALC_PRICE_UI: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::PLUGIN_PRM_CALC_PRICE: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::PLUGIN_PRM_CALC_PRICE_UI: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::PLUGIN_TCP_ZVT_CC: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::OPKG_COMMANDS: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::HARDWARE_DEVICES: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::OS: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_CASH_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_CONF_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_DEVICE_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT01_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT02_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT03_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT04_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT05_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT06_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT07_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT08_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT09_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT10_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT11_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT12_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT13_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT14_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT15_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT16_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT17_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT18_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT19_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT20_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT21_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT22_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT23_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT24_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT25_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT26_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT27_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT28_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT29_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT30_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT31_JSON: | ||||
|     break; | ||||
|     case UPDATE_COMPONENT::DC2C_PRINT32_JSON: | ||||
|     break; | ||||
|     } | ||||
| 
 | ||||
|     qInfo() << buf; | ||||
| 
 | ||||
|     return buf; | ||||
| } | ||||
| #endif | ||||
| 
 | ||||
| QString IsmasClient::updateOfPSAContinues(QString currentStage, | ||||
|                                           QString currentStageInfo, | ||||
|                                           QString const &version) { | ||||
| @@ -14,6 +14,7 @@ struct PSAInstalled { | ||||
|     } versionInfo; | ||||
| 
 | ||||
|     struct Tariff { | ||||
|         // VersionInfo versionInfo;
 | ||||
|         QString name; | ||||
|         QString version; | ||||
|         QString project; | ||||
| @@ -31,6 +32,7 @@ struct PSAInstalled { | ||||
|     } hw; | ||||
| 
 | ||||
|     struct Opkg { | ||||
|         // VersionInfo versionInfo;
 | ||||
|         int size; | ||||
|         QString blob; | ||||
|         QString lastCommit; | ||||
| @@ -38,6 +40,7 @@ struct PSAInstalled { | ||||
|     } opkg; | ||||
| 
 | ||||
|     struct DC { | ||||
|         // VersionInfo versionInfo;
 | ||||
|         QString versionHW; | ||||
|         QString versionSW; | ||||
|         QString gitBlob; | ||||
| @@ -62,6 +65,7 @@ struct PSAInstalled { | ||||
|     } pluginVersion; | ||||
| 
 | ||||
|     struct DC2C { | ||||
|         // VersionInfo versionInfo;
 | ||||
|         QString name; | ||||
|         QString blob; | ||||
|         int size; | ||||
| @@ -141,13 +145,47 @@ public: | ||||
|         DIRECT_PORT = 7778 | ||||
|     }; | ||||
| 
 | ||||
| 
 | ||||
|     enum RESULT_CODE { | ||||
|         SUCCESS=0, | ||||
|         // if between 00:00 - 04:00 Wait-button state not WAIT, then we assume
 | ||||
|         // that's an automatic nightly (not-necessary) update
 | ||||
|         NO_UPDATE_NECESSARY=1, | ||||
|         BACKUP_FAILED=2, | ||||
|         WRONG_PACKAGE=3, | ||||
|         INSTALL_ERROR=4}; | ||||
|         // if APISM reports the ISMAS is not available (15x, 6s delay each)
 | ||||
|         ISMAS_NO_CONNECTION_ERROR=2, | ||||
|         // if not within 00:00-04:00 and WAIT-button was not in state WAIT
 | ||||
|         ISMAS_TRIGGER_ERROR=3, | ||||
|         // cloning git repo. not possible
 | ||||
|         GIT_CLONE_ERROR=4, | ||||
|         // pulling from remote git server not possible
 | ||||
|         GIT_PULL_ERROR=5, | ||||
|         // fetching from remote git server not possible
 | ||||
|         GIT_FETCH_ERROR=6, | ||||
|         // merging fetched data not possible
 | ||||
|         GIT_MERGE_ERROR=7, | ||||
|         // check sanity of local customer-repository failed
 | ||||
|         GIT_CHECK_REPOSITORY_ERROR=8, | ||||
|         // switch/checkout of branch (i.e. zone) on error
 | ||||
|         GIT_SWITCH_BRANCH_ERROR=9, | ||||
|         // fetch/pull of new branch failed. the new branch was not available
 | ||||
|         // when installing via SD-card followed by intial clone during the
 | ||||
|         // update process.
 | ||||
|         GIT_FETCH_NEW_BRANCH_ERROR=10, | ||||
|         // error computing git-blob hash-value
 | ||||
|         GIT_HASH_ERROR=11, | ||||
|         // update for general json files failed.
 | ||||
|         JSON_FILES_UPDATE_ERROR=12, | ||||
|         // error downloading config-json-files to device controller
 | ||||
|         JSON_FILES_DOWNLOAD_ERROR=13, | ||||
|         // error downloading device-controller
 | ||||
|         DC_DOWNLOAD_ERROR=14, | ||||
|         // error rsyncing json/ini-files to local filesystem
 | ||||
|         RSYN_ERROR=15, | ||||
|         // HASH_VALUE_ERROR=14,
 | ||||
|         // HW_COMPATIBILITY_ERROR=15,
 | ||||
|         OPKG_COMMANDS_ERROR=16, | ||||
|         // CLEANUP_ERROR=18,
 | ||||
|         UPDATE_IN_ERROR_STATE=99 | ||||
|     }; | ||||
| 
 | ||||
|     enum REASON { | ||||
|         TIME_TRIGGERED = 0, | ||||
| @@ -194,8 +232,68 @@ public: | ||||
|     QString jsonParseFailed(int resultCode, QString reason, QString const &version = QString()); | ||||
|     std::optional<QString> finalResult(int resultCode, QString reason, QString const &version = QString()); | ||||
| 
 | ||||
|     // legacy
 | ||||
|     QString updateOfPSASendVersion(PSAInstalled const &psa); | ||||
| 
 | ||||
| #if 0 | ||||
|     enum class UPDATE_COMPONENT { | ||||
|         TARIFF, | ||||
|         SOFTWARE_ATBQT, | ||||
|         SOFTWARE_APISM, | ||||
|         SOFTWARE_ATB_UPDATE_TOOL, | ||||
|         CONFIG_PTU5_CPU_SERIAL, | ||||
|         CONFIG_DEVICE_CONTROLLER, | ||||
|         CONFIG_PRINTER, | ||||
|         CONFIG_BNA, | ||||
|         PLUGIN_ATB_DEVICE_CONTROLLER, | ||||
|         PLUGIN_INGENICO_CC, | ||||
|         PLUGIN_MOBILISIS_CALC_PRICE, | ||||
|         PLUGIN_MOBILISIS_CALC_PRICE_UI, | ||||
|         PLUGIN_PRM_CALC_PRICE, | ||||
|         PLUGIN_PRM_CALC_PRICE_UI, | ||||
|         PLUGIN_TCP_ZVT_CC, | ||||
|         OPKG_COMMANDS, | ||||
|         HARDWARE_DEVICES, | ||||
|         OS, | ||||
|         DC2C_CASH_JSON, | ||||
|         DC2C_CONF_JSON, | ||||
|         DC2C_DEVICE_JSON, | ||||
|         DC2C_PRINT01_JSON, | ||||
|         DC2C_PRINT02_JSON, | ||||
|         DC2C_PRINT03_JSON, | ||||
|         DC2C_PRINT04_JSON, | ||||
|         DC2C_PRINT05_JSON, | ||||
|         DC2C_PRINT06_JSON, | ||||
|         DC2C_PRINT07_JSON, | ||||
|         DC2C_PRINT08_JSON, | ||||
|         DC2C_PRINT09_JSON, | ||||
|         DC2C_PRINT10_JSON, | ||||
|         DC2C_PRINT11_JSON, | ||||
|         DC2C_PRINT12_JSON, | ||||
|         DC2C_PRINT13_JSON, | ||||
|         DC2C_PRINT14_JSON, | ||||
|         DC2C_PRINT15_JSON, | ||||
|         DC2C_PRINT16_JSON, | ||||
|         DC2C_PRINT17_JSON, | ||||
|         DC2C_PRINT18_JSON, | ||||
|         DC2C_PRINT19_JSON, | ||||
|         DC2C_PRINT20_JSON, | ||||
|         DC2C_PRINT21_JSON, | ||||
|         DC2C_PRINT22_JSON, | ||||
|         DC2C_PRINT23_JSON, | ||||
|         DC2C_PRINT24_JSON, | ||||
|         DC2C_PRINT25_JSON, | ||||
|         DC2C_PRINT26_JSON, | ||||
|         DC2C_PRINT27_JSON, | ||||
|         DC2C_PRINT28_JSON, | ||||
|         DC2C_PRINT29_JSON, | ||||
|         DC2C_PRINT30_JSON, | ||||
|         DC2C_PRINT31_JSON, | ||||
|         DC2C_PRINT32_JSON, | ||||
|     }; | ||||
|     QString sendLastVersion(UPDATE_COMPONENT comp, PSAInstalled const &psa); | ||||
| #endif | ||||
| 
 | ||||
|   private: | ||||
|     static void printDebugMessage(int port, QString const &clientIP, int clientPort, | ||||
|                                   QString const &message); | ||||
| @@ -9,7 +9,7 @@ | ||||
| #endif | ||||
| 
 | ||||
| #include "message_handler.h" | ||||
| #include "plugins/interfaces.h" | ||||
| #include <DeviceController/interfaces.h> | ||||
| #include "commandline_parser.h" | ||||
| 
 | ||||
| #include <unistd.h> | ||||
| @@ -74,14 +74,23 @@ int main(int argc, char *argv[]) { | ||||
|     QString plugInDir = parser.plugInDir(); | ||||
|     QString plugInName = parser.plugInName(); | ||||
|     QString workingDir = parser.workingDir(); | ||||
|     QString iniFileName = parser.iniFileName(); | ||||
|     bool const dryRun = parser.dryRun(); | ||||
|     bool const noUpdatePsaHardware = parser.noUpdatePsaHardware(); | ||||
|     bool const showYoctoVersion = parser.yoctoVersion(); | ||||
|     bool const showYoctoInstallStatus = parser.yoctoInstallStatus(); | ||||
|     bool const showExtendedVersion = parser.extendedVersion(); | ||||
|     bool const alwaysDownloadConfig = parser.alwaysDownloadConfig(); | ||||
|     bool const alwaysDownloadDC = parser.alwaysDownloadDC(); | ||||
| 
 | ||||
|     QString const rtPath = QCoreApplication::applicationDirPath(); | ||||
| 
 | ||||
|     int const machineNr = Utils::read1stLineOfFile("/mnt/system_data/machine_nr"); | ||||
|     int const customerNr = Utils::read1stLineOfFile("/mnt/system_data/cust_nr"); | ||||
|     int const zoneNr = Utils::read1stLineOfFile("/mnt/system_data/zone_nr"); | ||||
|     QString const branchName = (zoneNr != 0) | ||||
|             ? QString("zg1/zone%1").arg(zoneNr) : "master"; | ||||
| 
 | ||||
|     qInfo() << "pwd ......................" << rtPath; | ||||
|     qInfo() << "repositoryUrl ............" << repositoryUrl; | ||||
|     qInfo() << "plugInDir ................" << plugInDir; | ||||
| @@ -89,7 +98,16 @@ int main(int argc, char *argv[]) { | ||||
|     qInfo() << "workingDir ..............." << workingDir; | ||||
|     qInfo() << "dryRun ..................." << dryRun; | ||||
|     qInfo() << "noUpdatePsaHardware ......" << noUpdatePsaHardware; | ||||
|     qInfo() << "alwaysDownloadConfig ....." << alwaysDownloadConfig; | ||||
|     qInfo() << "alwaysDownloadDC ........." << alwaysDownloadDC; | ||||
|     qInfo() << "showYoctoVersion ........." << showYoctoVersion; | ||||
|     qInfo() << "showYoctoInstallStatus ..." << showYoctoInstallStatus; | ||||
|     qInfo() << "showExtendedVersion ......" << showExtendedVersion; | ||||
|     qInfo() << "iniFileName .............." << iniFileName; | ||||
|     qInfo() << "extended-version ........." << APP_EXTENDED_VERSION; | ||||
|     qInfo() << "machineNr ................" << machineNr; | ||||
|     qInfo() << "customerNr ..............." << customerNr; | ||||
|     qInfo() << "zoneNr ..................." << zoneNr; | ||||
| 
 | ||||
|     if (showExtendedVersion) { | ||||
|         printf(APP_EXTENDED_VERSION"\n"); | ||||
| @@ -104,11 +122,6 @@ int main(int argc, char *argv[]) { | ||||
|         return 0; | ||||
|     } | ||||
| 
 | ||||
|     int machineNr = Utils::read1stLineOfFile("/etc/machine_nr"); | ||||
|     int customerNr = Utils::read1stLineOfFile("/etc/cust_nr"); | ||||
|     int zoneNr = Utils::read1stLineOfFile("/etc/zone_nr"); | ||||
|     QString const branchName = (zoneNr != 0) | ||||
|             ? QString("zg1/zone%1").arg(zoneNr) : "master"; | ||||
| 
 | ||||
|     QThread::currentThread()->setObjectName("main thread"); | ||||
|     qInfo() << "Main thread" << QThread::currentThreadId(); | ||||
| @@ -117,8 +130,10 @@ int main(int argc, char *argv[]) { | ||||
|     if (!QDir(plugInDir).exists()) { | ||||
|         qCritical() << plugInDir | ||||
|                     << "does not exists, but has to contain dc-library"; | ||||
|         if (noUpdatePsaHardware == false) { | ||||
|             exit(-1); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     // before loading the library, delete all possible shared memory segments
 | ||||
| #if defined Q_OS_LINUX || defined Q_OS_UNIX | ||||
| @@ -136,6 +151,8 @@ int main(int argc, char *argv[]) { | ||||
|                   plugInName, | ||||
|                   workingDir, | ||||
|                   noUpdatePsaHardware, | ||||
|                   alwaysDownloadConfig, | ||||
|                   alwaysDownloadDC, | ||||
|                   dryRun); | ||||
| 
 | ||||
|     MainWindow mw(&worker); | ||||
| @@ -4,7 +4,7 @@ | ||||
| #include "utils.h" | ||||
| #include "progress_event.h" | ||||
| #include "update_dc_event.h" | ||||
| #include "plugins/interfaces.h" | ||||
| #include <DeviceController/interfaces.h> | ||||
| 
 | ||||
| #include <QDateTime> | ||||
| #include <QMessageBox> | ||||
| @@ -146,7 +146,7 @@ void MainWindow::onEnableExit() { | ||||
| 
 | ||||
| void MainWindow::onRestartExitTimer() { | ||||
|     m_exitTimer->stop(); | ||||
|     m_exitTimer->start(60 * 1000); | ||||
|     m_exitTimer->start(5 * 1000); | ||||
| 
 | ||||
|     scrollDownTextEdit(); | ||||
|     ui->updateStatus->setEnabled(false); | ||||
| @@ -208,13 +208,12 @@ void MainWindow::onAppendText(QString text, QString suffix) { | ||||
|     // Utils::printLineEditInfo(editText.split('\n', QString::SplitBehavior::SkipEmptyParts));
 | ||||
|     // ui->updateStatus->setText(editText.trimmed());
 | ||||
| 
 | ||||
|     // scrollDownTextEdit();
 | ||||
|     scrollDownTextEdit(); | ||||
| } | ||||
| 
 | ||||
| void MainWindow::onReplaceLast(QStringList newTextLines, QString suffix) { | ||||
|     // Utils::printInfoMsg(QString("ON REPLACE LAST (LIST) CALLED AT ")
 | ||||
|     //    + QDateTime::currentDateTime().toString(Qt::ISODateWithMs));
 | ||||
| 
 | ||||
|     int const s = newTextLines.size(); | ||||
|     if (s > 0) { | ||||
|         QString editText = ui->updateStatus->toPlainText(); | ||||
| @@ -249,11 +248,15 @@ void MainWindow::onReplaceLast(QStringList newTextLines, QString suffix) { | ||||
| void MainWindow::onReplaceLast(QString text, QString suffix) { | ||||
|     // Utils::printInfoMsg(QString("ON REPLACE LAST (TEXT) CALLED AT ")
 | ||||
|     //    + QDateTime::currentDateTime().toString(Qt::ISODateWithMs));
 | ||||
| 
 | ||||
|     QString editText = ui->updateStatus->toPlainText(); | ||||
|     QStringList lines = editText.split('\n', QString::SplitBehavior::SkipEmptyParts); | ||||
|     if (lines.size() > 0) { | ||||
|         // removing the last line is really meant for refreshing the last line
 | ||||
|         // with a string very similar than the original one, typically only
 | ||||
|         // followed by a suffix.
 | ||||
|         if (lines.last().contains(text)) { | ||||
|             lines.removeLast(); | ||||
|         } | ||||
|         if (!suffix.isNull() && suffix.size() > 0 && suffix != "\n") { | ||||
|             QString const add = text.leftJustified(m_width-10) + suffix; | ||||
|             if (!add.isEmpty()) { | ||||
							
								
								
									
										97
									
								
								UpdatePTUDevCtrl/message_handler.cpp
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										97
									
								
								UpdatePTUDevCtrl/message_handler.cpp
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,97 @@ | ||||
| #include "message_handler.h" | ||||
|  | ||||
| #include <QDateTime> | ||||
| #include <cstring> | ||||
| #include <QString> | ||||
| #include <QFileInfo> | ||||
| #include <QMessageLogContext> | ||||
|  | ||||
|  | ||||
| static char const *DBG_NAME[] = { "DBG  ", "WARN ", "CRIT ", "FATAL", "INFO " }; | ||||
| static bool installedMsgHandler = false; | ||||
| static int debugLevel = LOG_NOTICE; | ||||
|  | ||||
| int getDebugLevel() { return debugLevel; } | ||||
| void setDebugLevel(int newDebugLevel) { | ||||
|     debugLevel = newDebugLevel; | ||||
| } | ||||
|  | ||||
| bool messageHandlerInstalled() { | ||||
|     return installedMsgHandler; | ||||
| } | ||||
|  | ||||
| QtMessageHandler atbInstallMessageHandler(QtMessageHandler handler) { | ||||
|     installedMsgHandler = (handler != 0); | ||||
|     static QtMessageHandler prevHandler = nullptr; | ||||
|     if (handler) { | ||||
|         prevHandler = qInstallMessageHandler(handler); | ||||
|         return prevHandler; | ||||
|     } else { | ||||
|         return qInstallMessageHandler(prevHandler); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// | ||||
| /// \brief Print message according to given debug level. | ||||
| /// | ||||
| /// \note Install this function using qInstallMsgHandler(). | ||||
| /// | ||||
| ///  int main(int argc, char **argv) { | ||||
| ///     installMsgHandler(atbDebugOutput); | ||||
| ///     QApplication app(argc, argv); | ||||
| ///     ... | ||||
| ///     return app.exec(); | ||||
| ///  } | ||||
| /// | ||||
| #if (QT_VERSION > QT_VERSION_CHECK(5, 0, 0) && QT_VERSION < QT_VERSION_CHECK(6, 0, 0)) | ||||
| void atbDebugOutput(QtMsgType type, const QMessageLogContext &context, const QString &msg) { | ||||
|     Q_UNUSED(context); | ||||
|     QString const localMsg = QString(DBG_NAME[type]) + msg.toLocal8Bit(); | ||||
|  | ||||
|     switch (debugLevel) { | ||||
|         case LOG_DEBUG: { // debug-level message | ||||
|             syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|         } break; | ||||
|         case LOG_INFO: { // informational message | ||||
|             if (type != QtDebugMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         case LOG_NOTICE: { // normal, but significant, condition | ||||
|             if (type != QtDebugMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         case LOG_WARNING: { // warning conditions | ||||
|             if (type != QtInfoMsg && type != QtDebugMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         case LOG_ERR: { // error conditions | ||||
|             if (type != QtInfoMsg && type != QtDebugMsg && type != QtWarningMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         case LOG_CRIT: { // critical conditions | ||||
|             if (type != QtInfoMsg && type != QtDebugMsg && type != QtWarningMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         case LOG_ALERT: { // action must be taken immediately | ||||
|             if (type != QtInfoMsg && type != QtDebugMsg && type != QtWarningMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         case LOG_EMERG: { // system is unusable | ||||
|             if (type != QtInfoMsg && type != QtDebugMsg && type != QtWarningMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         default: { | ||||
|             //fprintf(stderr, "%s No ErrorLevel defined! %s\n", | ||||
|             //    datetime.toStdString().c_str(), msg.toStdString().c_str()); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| #endif | ||||
|  | ||||
							
								
								
									
										23
									
								
								UpdatePTUDevCtrl/message_handler.h
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										23
									
								
								UpdatePTUDevCtrl/message_handler.h
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| #ifndef MESSAGE_HANDLER_H_INCLUDED | ||||
| #define MESSAGE_HANDLER_H_INCLUDED | ||||
|  | ||||
| #include <QtGlobal> | ||||
| #ifdef __linux__ | ||||
| #include <syslog.h> | ||||
| #endif | ||||
|  | ||||
| int getDebugLevel(); | ||||
| void setDebugLevel(int newDebugLevel); | ||||
|  | ||||
| bool messageHandlerInstalled(); | ||||
| QtMessageHandler atbInstallMessageHandler(QtMessageHandler handler); | ||||
|  | ||||
| #if QT_VERSION < QT_VERSION_CHECK(5, 0, 0) | ||||
| // typedef void (*QtMessageHandler)(QtMsgType, const char *); | ||||
| void atbDebugOutput(QtMsgType type, const char *msg); | ||||
| #elif QT_VERSION < QT_VERSION_CHECK(6, 0, 0) | ||||
| // typedef void (*QtMessageHandler)(QtMsgType, const QMessageLogContext &, const QString &); | ||||
| void atbDebugOutput(QtMsgType type, const QMessageLogContext &context, const QString &msg); | ||||
| #endif | ||||
|  | ||||
| #endif // MESSAGE_HANDLER_H_INCLUDED | ||||
							
								
								
									
										124
									
								
								UpdatePTUDevCtrl/process/command.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										124
									
								
								UpdatePTUDevCtrl/process/command.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,124 @@ | ||||
| #include "command.h" | ||||
|  | ||||
| #include <QProcess> | ||||
| #include <QDebug> | ||||
| #include <QDir> | ||||
| #include <QRegularExpression> | ||||
| #include <QDateTime> | ||||
|  | ||||
| Command::Command(QString const &command, int start_timeout, int finish_timeout) | ||||
|     : m_command(command.trimmed()) | ||||
|     , m_commandResult("") | ||||
|     , m_waitForStartTimeout(start_timeout) | ||||
|     , m_waitForFinishTimeout(finish_timeout) | ||||
|     , m_exitCode(-1) { | ||||
| } | ||||
|  | ||||
| QString Command::getCommandResult() const { | ||||
|     return m_commandResult; | ||||
| } | ||||
|  | ||||
| void Command::readyReadStandardOutput() { | ||||
|     QProcess *p = (QProcess *)sender(); | ||||
|     m_commandResult += p->readAllStandardOutput(); | ||||
|     // qCritical() << m_commandResult; | ||||
| } | ||||
|  | ||||
| void Command::readyReadStandardError() { | ||||
|     QProcess *p = (QProcess *)sender(); | ||||
|     QByteArray buf = p->readAllStandardError(); | ||||
|     qCritical() << buf; | ||||
| } | ||||
|  | ||||
| void Command::finished(int /*exitCode*/, QProcess::ExitStatus /*exitStatus*/) { | ||||
|     QProcess *p = (QProcess *)sender(); | ||||
|     // read all remaining data sent to the process, just in case | ||||
|     QString d = p->readAllStandardOutput(); | ||||
|     if (!d.isEmpty()) { | ||||
|         m_commandResult += d; | ||||
|     } | ||||
|     disconnect(p, SIGNAL(finished(int,QProcess::ExitStatus)), this, SLOT(readyReadStandardOutput())); | ||||
|     disconnect(p, SIGNAL(finished(int,QProcess::ExitStatus)), this, SLOT(readyReadStandardError())); | ||||
| } | ||||
|  | ||||
| bool Command::execute(QString workingDirectory, QStringList args) { | ||||
|  | ||||
|     if (!QDir::setCurrent(workingDirectory)) { | ||||
|         qCritical() << "SET WORKING_DIRECTORY" << workingDirectory | ||||
|                     << "FAILED FOR" << m_command; | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     QScopedPointer<QProcess> p(new QProcess(this)); | ||||
|     p->setWorkingDirectory(workingDirectory); | ||||
|     p->setProcessChannelMode(QProcess::MergedChannels); | ||||
|  | ||||
|     connect(&(*p), SIGNAL(readyReadStandardOutput()), this, SLOT(readyReadStandardOutput())); | ||||
|     connect(&(*p), SIGNAL(readyReadStandardError()), this, SLOT(readyReadStandardError())); | ||||
|  | ||||
|     if (!args.isEmpty()) { | ||||
|         qDebug() << "START COMMAND" << m_command << "WITH ARGS" << args | ||||
|                  << "IN" << p->workingDirectory(); | ||||
|         p->start(m_command, args); | ||||
|     } else { | ||||
|         qDebug() << "START COMMAND" << m_command | ||||
|                  << "IN" << p->workingDirectory(); | ||||
|         p->start(m_command); | ||||
|     } | ||||
|  | ||||
|     qint64 const start = QDateTime::currentDateTime().toMSecsSinceEpoch(); | ||||
|  | ||||
|     if (p->waitForStarted(m_waitForStartTimeout)) { | ||||
|         qDebug() << "PROCESS" << m_command << "STARTED IN" << p->workingDirectory(); | ||||
|         if (p->state() == QProcess::ProcessState::Running) { | ||||
|             qDebug() << "PROCESS" << m_command << "RUNNING IN" << p->workingDirectory(); | ||||
|             // wait forever for git/opkg-commands to finish | ||||
|             int wait = m_waitForFinishTimeout; | ||||
|             if (m_command.trimmed().startsWith("git", Qt::CaseInsensitive) || | ||||
|                 m_command.trimmed().startsWith("opkg", Qt::CaseInsensitive)) { | ||||
|                 wait = -1; | ||||
|             } | ||||
|             bool const no_timeout = p->waitForFinished(wait); | ||||
|             if (no_timeout) { | ||||
|                 qDebug() << "PROCESS" << m_command << "FINISHED IN" << p->workingDirectory(); | ||||
|                 if (p->exitStatus() == QProcess::NormalExit) { | ||||
|                     if ((m_exitCode = p->exitCode()) == 0) { | ||||
|                         qint64 const end = QDateTime::currentDateTime().toMSecsSinceEpoch(); | ||||
|                         qDebug() << "EXECUTED" << m_command | ||||
|                                  << QString("(runtime %1ms)").arg(end-start) | ||||
|                                  << "with code" << m_exitCode | ||||
|                                  << "IN" << p->workingDirectory(); | ||||
|                         return true; | ||||
|                     } else { | ||||
|                         qint64 const end = QDateTime::currentDateTime().toMSecsSinceEpoch(); | ||||
|                         qCritical() << "EXECUTED" << m_command | ||||
|                                     << QString("(runtime %1ms)").arg(end-start) | ||||
|                                     << "with code" << m_exitCode | ||||
|                                     << "IN" << p->workingDirectory(); | ||||
|                     } | ||||
|                 } else { | ||||
|                     qint64 const end = QDateTime::currentDateTime().toMSecsSinceEpoch(); | ||||
|                     qCritical() << "PROCESS" << m_command << "CRASHED with code" | ||||
|                                 << p->exitCode() | ||||
|                                 << QString("(after %1ms)").arg(end-start) | ||||
|                                 << "IN" << p->workingDirectory(); | ||||
|                 } | ||||
|             } else { | ||||
|                 qint64 const end = QDateTime::currentDateTime().toMSecsSinceEpoch(); | ||||
|                 qCritical() << "PROCESS" << m_command | ||||
|                             << "DID NOT FINISH WITH" << wait | ||||
|                             << "MS IN" << p->workingDirectory() | ||||
|                             << QString("(runtime %1ms)").arg(end-start); | ||||
|             } | ||||
|         } else { | ||||
|             qCritical() << "WRONG PROCESS STATE" << p->state() | ||||
|                         << "IN" << p->workingDirectory(); | ||||
|         } | ||||
|     } else { | ||||
|         qint64 const end = QDateTime::currentDateTime().toMSecsSinceEpoch(); | ||||
|         qCritical() << "PROCESS" << m_command << "TIMEOUT AT START" | ||||
|                     << QString("(runtime %1ms)").arg(end-start) | ||||
|                     << "IN" << p->workingDirectory(); | ||||
|     } | ||||
|     return false; | ||||
| } | ||||
							
								
								
									
										869
									
								
								UpdatePTUDevCtrl/update.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										869
									
								
								UpdatePTUDevCtrl/update.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,869 @@ | ||||
| #include "update.h" | ||||
| #include "worker.h" | ||||
| #include "utils.h" | ||||
| #include "update_dc_event.h" | ||||
| #include "mainwindow.h" | ||||
|  | ||||
| #include <QCoreApplication> | ||||
| #include <QApplication> | ||||
| #include <QFile> | ||||
| #include <QTemporaryFile> | ||||
| #include <QDebug> | ||||
| #include <QTextStream> | ||||
| #include <QRegularExpression> | ||||
| #include <QRegExp> | ||||
| #include <QApplication> | ||||
|  | ||||
| #if defined (Q_OS_UNIX) || defined (Q_OS_LINUX) | ||||
| #include "unistd.h" | ||||
| #endif | ||||
|  | ||||
| #include <DeviceController/interfaces.h> | ||||
|  | ||||
| #include <QSharedMemory> | ||||
| #include <QScopedPointer> | ||||
| #include <QDir> | ||||
| #include <QThread> | ||||
| #include <QDateTime> | ||||
| #include <QPluginLoader> | ||||
| #include <QMap> | ||||
|  | ||||
| #define UPDATE_OPKG                 (1) | ||||
| #define UPDATE_DC                   (0) | ||||
|  | ||||
| static const QMap<QString, int> baudrateMap = { | ||||
|   {"1200"   ,   0}, {"9600"   ,   1}, {"19200"  ,   2}, {"38400"  ,   3}, | ||||
|   {"57600"  ,   4}, {"115200" ,   5} | ||||
| }; | ||||
|  | ||||
| QPluginLoader Update::pluginLoader; | ||||
|  | ||||
| hwinf *Update::loadDCPlugin(QDir const &plugInDir, QString const &fname) { | ||||
|     hwinf *hw = nullptr; | ||||
|     if (plugInDir.exists()) { | ||||
|         QString pluginLibName(fname); | ||||
|         pluginLibName = plugInDir.absoluteFilePath(pluginLibName); | ||||
|         QFileInfo info(pluginLibName); | ||||
|         if (info.exists()) { | ||||
|             pluginLibName = plugInDir.absoluteFilePath(pluginLibName); | ||||
|             pluginLoader.setFileName(pluginLibName); | ||||
|             // static QPluginLoader pluginLoader(pluginLibName); | ||||
|             if (!pluginLoader.load()) { | ||||
|                 qCritical() << "in directory" << plugInDir.absolutePath(); | ||||
|                 qCritical() << "cannot load plugin" << pluginLoader.fileName(); | ||||
|                 qCritical() << pluginLoader.errorString(); | ||||
|             } | ||||
|  | ||||
|             qCritical() << "loadDCPlugin() plugin directory:" << plugInDir.absolutePath(); | ||||
|             qCritical() << "loadDCPlugin() plugin file name:" << pluginLoader.fileName(); | ||||
|  | ||||
|             if (!pluginLoader.isLoaded()) { | ||||
|                 qCritical() << pluginLoader.errorString(); | ||||
|             } | ||||
|             QObject *plugin = pluginLoader.instance(); | ||||
|             if (!plugin) { | ||||
|                 qCritical() << "cannot start instance"; | ||||
|             } | ||||
|             if (! (hw = qobject_cast<hwinf *>(plugin))) { | ||||
|                 qCritical() << "cannot cast plugin" << plugin << "to hwinf"; | ||||
|             } | ||||
|         } else { | ||||
|             qCritical() << pluginLibName << "does not exist"; | ||||
|         } | ||||
|     } else { | ||||
|         qCritical() << "plugins directory" << plugInDir.absolutePath() | ||||
|                     << "does not exist"; | ||||
|     } | ||||
|     return hw; | ||||
| } | ||||
|  | ||||
| bool Update::unloadDCPlugin() { | ||||
|     if (pluginLoader.unload()) { | ||||
|         qCritical() << "unloaded plugin" << pluginLoader.fileName(); | ||||
|         // Note: will re-instantiate the library ! | ||||
|         // QObject *rootObject = pluginLoader.instance(); | ||||
|         // if (rootObject) { | ||||
|         //    qCritical() << "reloaded plugin: root object again available"; | ||||
|         //    return false; | ||||
|         // } | ||||
|         // qCritical()unloaded plugin: root object gone"; | ||||
|         return true; | ||||
|     } | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| class hwapi; | ||||
| Update::Update(Worker *worker, | ||||
|                QString customerRepository, | ||||
|                QString customerNrStr, | ||||
|                QString branchName, | ||||
|                QString plugInDir, | ||||
|                QString pluginName, | ||||
|                QString workingDir, | ||||
|                bool dryRun, | ||||
|                QObject *parent, | ||||
|                char const *serialInterface, | ||||
|                char const *baudrate) | ||||
|     : QObject(parent) | ||||
|     , m_hw(loadDCPlugin(QDir(plugInDir), pluginName)) | ||||
|     , m_worker(worker) | ||||
|     , m_serialInterface(serialInterface) | ||||
|     , m_baudrate(baudrate) | ||||
|     , m_customerRepository(customerRepository) | ||||
|     , m_customerNrStr(customerNrStr) | ||||
|     , m_branchName(branchName) | ||||
|     , m_pluginName(pluginName) | ||||
|     , m_workingDir(workingDir) | ||||
|     , m_dryRun(dryRun) | ||||
|     , m_sys_areDCdataValid(false) { | ||||
|  | ||||
|     if (!m_hw) { | ||||
|         qCritical() << "(" << __func__ << ":" << __LINE__ << ") m_hw == nullptr -> ca-slave plugin not loaded"; | ||||
|     } else { | ||||
|         int tries = 20; | ||||
|         while ((m_sys_areDCdataValid = m_hw->sys_areDCdataValid()) == false) { | ||||
|             // must deliver 'true', only then are all data from hwapi valid | ||||
|             if (--tries < 0) { | ||||
|                 qCritical() << "ERROR!!! DC DATA NOT VALID -> CA-MASTER-PLUGIN NOT CONNECTED"; | ||||
|                 break; | ||||
|             } | ||||
|             m_hw->dc_autoRequest(true); | ||||
|             QThread::msleep(500); | ||||
|         } | ||||
|  | ||||
|         qCritical() << "(" << __func__ << ":" << __LINE__ << ") m_sys_areDCDataValid ..." | ||||
|                     << m_sys_areDCdataValid; | ||||
|  | ||||
| #if 0 | ||||
|         QObject const *obj = m_hw->getAPI(); | ||||
|         Q_ASSERT(obj != nullptr); | ||||
|  | ||||
|         QDebug critical = qCritical(); | ||||
|         critical << "connect() to onReportDCDownloadStatus() ..."; | ||||
|         if (!connect(obj, | ||||
|                      SIGNAL(hwapi_reportDCDownloadStatus(QString const&)), | ||||
|                      this, | ||||
|                      SLOT(onReportDCDownloadStatus(QString const &)))) { | ||||
|             critical << "FAILED"; | ||||
|         } else critical << "DONE"; | ||||
|  | ||||
|         critical = qCritical(); | ||||
|         critical << "connect() to onReportDCDownloadSuccess() ..."; | ||||
|         if (!connect(obj, | ||||
|             SIGNAL(hwapi_reportDCDownloadSuccess(QString const&)), this, | ||||
|             SLOT(onReportDCDownloadSuccess(QString const &)))) { | ||||
|             critical << "FAILED"; | ||||
|         } else critical << "DONE"; | ||||
|  | ||||
|         critical = qCritical(); | ||||
|         critical << "connect() to onReportDCDownloadFailure() ..."; | ||||
|         if (!connect(obj, | ||||
|             SIGNAL(hwapi_reportDCDownloadFailure(QString const &)), this, | ||||
|             SLOT(onReportDCDownloadFailure(QString const &)))) { | ||||
|             critical << "FAILED"; | ||||
|         } else critical << "DONE"; | ||||
| #endif | ||||
|     } | ||||
| } | ||||
|  | ||||
| Update::~Update() { | ||||
| } | ||||
|  | ||||
| void Update::onReportDCDownloadStatus(QString const &status) { | ||||
|     emit m_worker->showStatusMessage("DL", status); | ||||
| } | ||||
|  | ||||
| void Update::onReportDCDownloadSuccess(QString const &msg) { | ||||
|     qCritical() << "msg" << msg; | ||||
| } | ||||
|  | ||||
| void Update::onReportDCDownloadFailure(QString const &errorMsg) { | ||||
|     qCritical() << "msg" << errorMsg; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| // br is a index into a table, used for historical reasons. | ||||
| bool Update::openSerial(int br, QString baudrate, QString comPort) const { | ||||
|     if (m_hw) { | ||||
|         qDebug() << "opening serial" << br << baudrate << comPort << "..."; | ||||
|         if (m_hw->dc_openSerial(br, baudrate, comPort, 1) == true) { // 1 for connect | ||||
|             Utils::printInfoMsg( | ||||
|                 QString("OPENING SERIAL %1").arg(br) | ||||
|                       + " " + baudrate + " " + comPort + "...OK"); | ||||
|  | ||||
|             // m_hw->dc_autoRequest(true); | ||||
|             m_hw->dc_autoRequest(false); | ||||
|             QThread::sleep(1); | ||||
|  | ||||
|             Utils::printInfoMsg(QString("IS PORT OPEN %1").arg(m_hw->dc_isPortOpen())); | ||||
|             return true; | ||||
|         } | ||||
|  | ||||
|         Utils::printCriticalErrorMsg( | ||||
|             QString("OPENING SERIAL %1").arg(br) | ||||
|                   + " " + baudrate + " " + comPort + "...FAILED"); | ||||
|     } | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| void Update::closeSerial() const { | ||||
|     qInfo() << "CLOSED SERIAL" << m_baudrate << m_serialInterface; | ||||
|     if (m_hw) { | ||||
|         m_hw->dc_closeSerial(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| bool Update::isSerialOpen() const { | ||||
|     return m_hw ? m_hw->dc_isPortOpen() : false; | ||||
| } | ||||
|  | ||||
| /* | ||||
|  | ||||
|  /////////////////////////////////////////////////////////////////////////////// | ||||
|  // | ||||
|  //                        USING THE DC BOOTLOADER | ||||
|  // | ||||
|  /////////////////////////////////////////////////////////////////////////////// | ||||
|  | ||||
|  1  :   bl_reboot()     // send to application, want DC2 to reset (in order to | ||||
|                         // start the bootloader) | ||||
|                         // | ||||
|                         // NOTE: this function is NOT reliable !!! Sometimes it | ||||
|                         // simply does not work, in which case bl_startBL, | ||||
|                         // bl_checkBL and bl_isUp do not work as well. | ||||
|                         // Alas, there is no feedback if bl_reboot worked! | ||||
|                         // | ||||
|                         // NOTE: this function can be called only once per | ||||
|                         // minute, because once called again, the controller | ||||
|                         // performs some self-checks consuming some time. | ||||
|                         // | ||||
|                         // NOTE: after a successful bl_reboot(), the device is | ||||
|                         // waiting about 4 seconds in the bootloader. To stay in | ||||
|                         // the bootloader, we have to send the command | ||||
|                         // bl_startBL(), which is kind of a misnomer, as it | ||||
|                         // should be bl_doNotLeaveBL(). | ||||
|                         // | ||||
|  2  :   bl_startBL():   // send within 4s after DC power-on, otherwise | ||||
|                         // bootloader is left. | ||||
|                         // | ||||
|                         // NOTE: a running bootloader is a MUST for the download | ||||
|                         // process of a device controller firmware as it does | ||||
|                         // the actual writing of the memory (the bl_reboot() | ||||
|                         // from above erases the available memory). | ||||
|                         // | ||||
|  3  :   bl_check():     // send command to verify if bl is up | ||||
|                         // | ||||
|                         // NOTE: this command is kind of a request that we want | ||||
|                         // to check if the bootloader is up. The device | ||||
|                         // (actually the bootloader) responds with its version. | ||||
|                         // | ||||
|  4  :   bl_isUp():      // returns true if bl is up and running | ||||
|                         // | ||||
|                         // NOTE: we know what the bootloader version actually is | ||||
|                         // as the bootloader does not change. By comparing the | ||||
|                         // string received in the previous step with this known | ||||
|                         // version string we know if the bootloader is up. | ||||
|                         // | ||||
|                         // NOTE FOR ALL PREVIOUS STEPS: execute them in their | ||||
|                         // own slots each to be sure to receive any possible | ||||
|                         // responds from the device. | ||||
|                         // | ||||
|  5  :   bl_sendAddress(blockNumber) | ||||
|                         // send start address, nr of 64-byte block, start with 0 | ||||
|                         // will be sent only for following block-numbers: | ||||
|                         // 0, 1024, 2048, 3072 and 4096, so basically every | ||||
|                         // 64kByte. | ||||
|                         // for other addresses nothing happens | ||||
|  | ||||
|  6  :   bl_wasSendingAddOK() | ||||
|                         // return val:  0: no response by now | ||||
|                         //              1: error | ||||
|                         //             10: OK | ||||
|  | ||||
|  7  :   bl_sendDataBlock() | ||||
|                         // send 64 byte from bin file | ||||
|  | ||||
|  8  :   bl_sendLastBlock() | ||||
|                         // send this command after all data are transferred | ||||
|  | ||||
|  9  :   bl_wasSendingDataOK() | ||||
|                         // return val:  0: no response by now | ||||
|                         //              1: error | ||||
|                         //             10: OK | ||||
|  | ||||
|  10 :   bl_stopBL()     // leave bl and start (the new) application | ||||
|                         // | ||||
|                         // NOTE: this function MUST work under all conditions. | ||||
|                         // Alas, there is no direct result for this command, so | ||||
|                         // the only way of knowing it was successful is to ask | ||||
|                         // the device if the bootloader is still running. | ||||
|                         // There is no problem to repeat this command until the | ||||
|                         // bootloader is really not running anymore. | ||||
|  */ | ||||
| bool Update::updateBinary(QString const &fileToSendToDC) { | ||||
|     qInfo() << "UPDATING DEVICE CONTROLLER FIRMWARE BINARY" << fileToSendToDC; | ||||
|  | ||||
|     return false; | ||||
|  | ||||
| #if 0 | ||||
|     QFile fn(fileToSendToDC); | ||||
|     if (!fn.exists()) { | ||||
|         // output via CONSOLE() etc | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     bool bl_isUp = false; | ||||
|     if (m_hw->bl_completeStart()) { | ||||
|         int cnt = 5; | ||||
|         while (--cnt > 0) { | ||||
|             if (m_hw->bl_isUp()) { | ||||
|                 bl_isUp = true; | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (!bl_isUp) { | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     if (!m_hw->bl_storeFirmware(fileToSendToDC)) { | ||||
|         m_hw->bl_stopBL(); | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     uint16_t const nrOfFirmwareBlocks = m_hw->bl_getNrOfFirmwareBlocks(); | ||||
|  | ||||
|     for (uint16_t blockNr = 0; blockNr <= nrOfFirmwareBlocks; ++blockNr) { | ||||
|         m_hw->bl_blockAutoLoad(blockNr); | ||||
|  | ||||
|         int sleepTime = 0; | ||||
|         while (1) { | ||||
|             if (sleepTime > 1500) { | ||||
|                 m_hw->bl_stopBL(); | ||||
|                 return false; | ||||
|             } | ||||
|  | ||||
|             int8_t const r = m_hw->bl_blockAutoResponse(); | ||||
|  | ||||
|             // after every "bl_blockAutoLoad()" call this until response | ||||
|             // retval  0: wait   1: OK, blk was sent    2: OK, transfer complete | ||||
|             //         3: error despite repeating, cancel. probably bin file corrupted | ||||
|             //      Max duration: 3x no response from BL = 900ms | ||||
|  | ||||
|             switch(r) { | ||||
|             case 1: | ||||
|             /* fall through */ | ||||
|             case 2: | ||||
|                 sleepTime = 0; | ||||
|                 break; | ||||
|             case 0: { | ||||
|                 QThread::msleep(100); | ||||
|                 sleepTime += 100; | ||||
|             } break; | ||||
|             case 3: | ||||
|                 m_hw->bl_stopBL(); | ||||
|                 return false; | ||||
|             default: | ||||
|                 m_hw->bl_stopBL(); | ||||
|                 return false; // unknown error code | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         m_hw->bl_stopBL(); | ||||
|     } | ||||
|  | ||||
|     return true; | ||||
| #endif | ||||
| } | ||||
|  | ||||
| QString Update::jsonType(enum FileTypeJson type) { | ||||
|     switch (type) { | ||||
|     case FileTypeJson::CASH: return "CASH"; | ||||
|     case FileTypeJson::CONFIG: return "CONFIG"; | ||||
|     case FileTypeJson::PRINTER: return "PRINTER"; | ||||
|     case FileTypeJson::SERIAL: return "SERIAL"; | ||||
|     case FileTypeJson::DEVICE: return "DEVICE"; | ||||
|     case FileTypeJson::TIME: return "TIME"; | ||||
|     } | ||||
|     return "N/A"; | ||||
| } | ||||
|  | ||||
| bool Update::downloadJson(enum FileTypeJson type, | ||||
|                           int templateIdx, | ||||
|                           QString jsFileToSendToDC) const { | ||||
|  | ||||
|     bool ret = false; | ||||
|     if (m_hw) { | ||||
|         m_hw->dc_autoRequest(true); // downloading Json needs the AutoEmission flag | ||||
|         qDebug() << "SET AUTO-REQUEST=TRUE"; | ||||
|         QThread::sleep(1);  // make sure the auto-request flag is acknowledged | ||||
|  | ||||
|         QStringList lst; | ||||
|         bool ready = false; | ||||
|         int nTry = 25; | ||||
|         while ((ready = m_hw->sys_ready4sending()) == false) { | ||||
|             QThread::msleep(200); | ||||
|             if (--nTry <= 0) { | ||||
|                 QString msg("SYS NOT READY FOR SENDING AFTER 5 SECONDS"); | ||||
|                 Utils::printCriticalErrorMsg(msg); | ||||
|                 lst << msg; | ||||
|  | ||||
|                 if (m_worker) { | ||||
|                     m_worker->ISMAS(lst) << (m_worker->GUI(lst) << (m_worker->CONSOLE(lst) << Worker::UPDATE_STEP::DOWNLOAD_FILES_TO_PSA_HARDWARE)); | ||||
|                 } | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         QString msg; | ||||
|         lst.clear(); | ||||
|         if (ready) { | ||||
|             QFile file(jsFileToSendToDC); | ||||
|             QFileInfo fi(jsFileToSendToDC); // max. size of template file is 800 bytes | ||||
|             if (file.exists()) { | ||||
|                 if (file.open(QIODevice::ReadOnly)) { | ||||
|                     if (fi.size() > 0 && fi.size() <= 800) { | ||||
|                         QByteArray ba = file.readAll(); | ||||
|                         // kindOfFile: 1=config, 2=device, 3=cash, 4=serial, 5=time, 6=printer | ||||
|                         //      nrOfTemplate=1...32 if kindOfFile==6 | ||||
|                         //      content = content of the Json file, max 800byte ascii signs | ||||
|                         if (m_hw->sys_sendJsonFileToDc((uint8_t)(type), | ||||
|                                                        templateIdx, | ||||
|                                                        (uint8_t *)ba.data())) { | ||||
|  | ||||
|                             /* | ||||
|                              *  Note: the machine id is contained in DC2C_conf.json. | ||||
|                              *  The idea was to use this to check if the download of | ||||
|                              *  the json-file was correct. It did not work, as the | ||||
|                              *  update of the PSA (to reflect a change in the | ||||
|                              *  machine id) did not happen immediately. | ||||
|                              * | ||||
|                                 m_hw->dc_autoRequest(true); | ||||
|                                 QThread::msleep(500); | ||||
|  | ||||
|                                 // testing | ||||
|                                 m_hw->request_ReadbackMachineID(); | ||||
|                                 QThread::msleep(500); | ||||
|  | ||||
|                                 uint8_t data[64]; | ||||
|                                 memset(data, 0x00, sizeof(data)); | ||||
|                                 uint8_t length = 0; | ||||
|  | ||||
|                                 m_hw->readback_machineIDdata(&length, data); | ||||
|  | ||||
|                                 QThread::msleep(500); | ||||
|  | ||||
|                                 QByteArray ba((const char*)data, length); | ||||
|  | ||||
|                                 qCritical() << length << "MACHINE ID =" << ba.toHex(':'); | ||||
|                             */ | ||||
|  | ||||
|                             if (m_worker) { | ||||
|                                 m_worker->ISMAS(lst) << (m_worker->GUI(lst) << (m_worker->CONSOLE(lst) << Worker::UPDATE_STEP::DOWNLOAD_FILES_TO_PSA_HARDWARE)); | ||||
|                             } | ||||
|  | ||||
|                             ret = true; | ||||
|                         } else { | ||||
|                             msg = QString("ERROR SEND JSON-FILE %1 TO DC").arg(file.fileName()); | ||||
|                             Utils::printCriticalErrorMsg(msg); | ||||
|                             lst << msg; | ||||
|                             if (m_worker) { | ||||
|                                 m_worker->ISMAS(lst) <<  (m_worker->GUI(lst) << (m_worker->CONSOLE(lst) | ||||
|                                                      << Worker::UPDATE_STEP::DOWNLOAD_FILES_TO_PSA_HARDWARE)); | ||||
|                             } | ||||
|                         } | ||||
|                     } else { | ||||
|                         msg = QString("SIZE OF %1 TOO BIG (%2 BYTES)").arg(jsFileToSendToDC).arg(fi.size()); | ||||
|                         Utils::printCriticalErrorMsg(msg); | ||||
|                         lst << msg; | ||||
|                         if (m_worker) { | ||||
|                             m_worker->ISMAS(lst) <<  (m_worker->GUI(lst) << (m_worker->CONSOLE(lst) | ||||
|                                                  << Worker::UPDATE_STEP::DOWNLOAD_FILES_TO_PSA_HARDWARE)); | ||||
|                         } | ||||
|                     } | ||||
|                 } else { | ||||
|                     msg = QString("CAN NOT OPEN ") + jsFileToSendToDC + " FOR READING"; | ||||
|                     Utils::printCriticalErrorMsg(msg); | ||||
|                     lst << msg; | ||||
|  | ||||
|                     if (m_worker) { | ||||
|                         m_worker->ISMAS(lst) <<  (m_worker->GUI(lst) << (m_worker->CONSOLE(lst) | ||||
|                                              << Worker::UPDATE_STEP::DOWNLOAD_FILES_TO_PSA_HARDWARE)); | ||||
|                     } | ||||
|                 } | ||||
|             } else { | ||||
|                 msg = QString(jsFileToSendToDC) + " DOES NOT EXIST"; | ||||
|                 Utils::printCriticalErrorMsg(msg); | ||||
|                 lst << msg; | ||||
|  | ||||
|                 if (m_worker) { | ||||
|                     m_worker->ISMAS(lst) <<  (m_worker->GUI(lst) << (m_worker->CONSOLE(lst) | ||||
|                                          << Worker::UPDATE_STEP::DOWNLOAD_FILES_TO_PSA_HARDWARE)); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         m_hw->dc_autoRequest(false); | ||||
|         qDebug() << "SET AUTO-REQUEST=FALSE"; | ||||
|         QThread::sleep(1);  // make sure the auto-request flag is acknowledged | ||||
|     } | ||||
|  | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| bool Update::updatePrinterTemplate(int templateIdx, QString jsFile) const { | ||||
|     return downloadJson(FileTypeJson::PRINTER, templateIdx, jsFile); | ||||
| } | ||||
|  | ||||
| bool Update::updateConfig(QString jsFile) { | ||||
|     return downloadJson(FileTypeJson::CONFIG, 0, jsFile); | ||||
| } | ||||
|  | ||||
| bool Update::updateCashConf(QString jsFile) { | ||||
|     return downloadJson(FileTypeJson::CASH, 0, jsFile); | ||||
| } | ||||
|  | ||||
| bool Update::updateDeviceConf(QString jsFile) { | ||||
|     return downloadJson(FileTypeJson::DEVICE, 0, jsFile); | ||||
| } | ||||
|  | ||||
| QStringList Update::split(QString line, QChar sep) { | ||||
|     QStringList lst; | ||||
|     QString next; | ||||
|     int start = 0, end; | ||||
|  | ||||
|     while ((end = line.indexOf(sep, start)) != -1) { | ||||
|         next = line.mid(start, end - start).trimmed(); | ||||
|         lst << next; | ||||
|         start = end + 1; | ||||
|     } | ||||
|     next = line.mid(start, end - start).trimmed(); | ||||
|     lst << next; | ||||
|  | ||||
|     return lst; | ||||
| } | ||||
|  | ||||
| void Update::readyReadStandardOutput() { | ||||
|     QProcess *p = (QProcess *)sender(); | ||||
|     QByteArray buf = p->readAllStandardOutput(); | ||||
|     qCritical() << buf; | ||||
| } | ||||
|  | ||||
| void Update::readyReadStandardError() { | ||||
|     QProcess *p = (QProcess *)sender(); | ||||
|     QByteArray buf = p->readAllStandardError(); | ||||
|     qCritical() << buf; | ||||
| } | ||||
|  | ||||
| void Update::finished(int /*exitCode*/, QProcess::ExitStatus /*exitStatus*/) { | ||||
|     QProcess *p = (QProcess *)sender(); | ||||
|     disconnect(p, SIGNAL(finished(int,QProcess::ExitStatus)), this, SLOT(readyReadStandardOutput())); | ||||
|     disconnect(p, SIGNAL(finished(int,QProcess::ExitStatus)), this, SLOT(readyReadStandardError())); | ||||
| } | ||||
|  | ||||
| QStringList Update::getDcSoftAndHardWareVersion() { | ||||
|     if (m_hw) { | ||||
|         m_hw->dc_autoRequest(true); | ||||
|         QThread::sleep(1); // make sure the timer-slots are active | ||||
|  | ||||
|         for (int i=0; i < 3; ++i) { // send explicit reuests to get | ||||
|                                     // current SW/HW-versions | ||||
|             m_hw->request_DC2_SWversion(); | ||||
|             m_hw->request_DC2_HWversion(); | ||||
|             QThread::sleep(1); | ||||
|         } | ||||
|  | ||||
|         QString const &hwVersion = m_hw->dc_getHWversion().toLower().trimmed(); | ||||
|         QString const &swVersion = m_hw->dc_getSWversion().toLower().trimmed(); | ||||
|  | ||||
|         m_hw->dc_autoRequest(false); | ||||
|         QThread::sleep(1); // make sure the timer-slots are inactive | ||||
|  | ||||
|         if (!hwVersion.isEmpty() && !swVersion.isEmpty()) { | ||||
|             return QStringList() << hwVersion << swVersion; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return QStringList() << "DC HW-version not available" | ||||
|                          << "DC SW-version not available"; | ||||
| } | ||||
|  | ||||
| QString Update::getFileVersion(QString const& jsonFileName) { | ||||
|     // "version":"15.10.2023 14:55 02.00.06", | ||||
|     static const QRegularExpression re("^.*(\\\"version\\\":)(.*)$"); | ||||
|  | ||||
|     QString fileVersion; | ||||
|     QFile inputFile(jsonFileName); | ||||
|     if (inputFile.open(QIODevice::ReadOnly)) { | ||||
|        QTextStream in(&inputFile); | ||||
|        while (!in.atEnd()) { | ||||
|           QString line = in.readLine(); | ||||
|  | ||||
|           QRegularExpressionMatch match; | ||||
|           int idx = line.indexOf(re, 0, &match); | ||||
|           if (idx != -1) { | ||||
|               fileVersion = match.captured(match.lastCapturedIndex()); | ||||
|               break; | ||||
|           } | ||||
|        } | ||||
|        inputFile.close(); | ||||
|     } | ||||
|  | ||||
|     return fileVersion; | ||||
| } | ||||
|  | ||||
| bool Update::checkDownloadedJsonVersions(QStringList const& jsonFileNames) { | ||||
|  | ||||
|     for (QStringList::size_type i=0; i < jsonFileNames.size(); ++i) { | ||||
|  | ||||
|         uint8_t jsonNr = 0; | ||||
|         QFileInfo fInfo(jsonFileNames[i]); | ||||
|  | ||||
|         if (fInfo.fileName().endsWith("conf.json")) { | ||||
|             jsonNr = 1; | ||||
|         } else | ||||
|         if (fInfo.fileName().endsWith("device.json")) { | ||||
|             jsonNr = 2; | ||||
|         } else | ||||
|         if (fInfo.fileName().endsWith("cash.json")) { | ||||
|             jsonNr = 3; | ||||
|         } else { | ||||
|             QRegularExpressionMatch match; | ||||
|             static const QRegularExpression re("^(.*print)([0-3][0-9])\\.json\\s*$"); | ||||
|             int idx = fInfo.fileName().indexOf(re, 0, &match); | ||||
|             if (idx != -1) { | ||||
|                 QString captured = match.captured(match.lastCapturedIndex()); | ||||
|                 bool ok = false; | ||||
|                 int n = captured.toInt(&ok); | ||||
|                 if (ok) { | ||||
|                     jsonNr = n + 4; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (jsonNr != 0) { | ||||
| #if 0 | ||||
|             m_hw->sys_requestJsonVersions(jsonNr); | ||||
|             QThread::msleep(500); | ||||
|  | ||||
|             char buf[64]; | ||||
|             memset(buf, 0x00, sizeof(buf)); | ||||
|             m_hw->sys_getJsonVersions(jsonNr, buf); | ||||
|             buf[sizeof(buf)-1] = '\0'; | ||||
|  | ||||
|             QString const installedVersion(buf); | ||||
|             QString const fileVersion = getFileVersion(jsonFileNames[i]); | ||||
|  | ||||
|             qCritical() << "installed version:" << installedVersion; | ||||
|             qCritical() << "     file version:" << fileVersion; | ||||
|  | ||||
|             if (installedVersion == fileVersion) { | ||||
|  | ||||
|             } | ||||
| #endif | ||||
|  | ||||
|         } else { | ||||
|             qCritical() << "CANNOT FIND JSON-NR FOR" << jsonFileNames[i]; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return false; | ||||
| } | ||||
| bool Update::doUpdate(int &displayIndex, QStringList const &filesToWorkOn) { | ||||
|  | ||||
|     if (!m_hw) { | ||||
|         Utils::printInfoMsg("CA-PLUGIN NOT LOADED"); | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     int tries = 20; | ||||
|     while ((m_sys_areDCdataValid = m_hw->sys_areDCdataValid()) == false) { | ||||
|         // must deliver 'true', only then are all data from hwapi valid | ||||
|         if (--tries < 0) { | ||||
|             Utils::printCriticalErrorMsg("ERROR!!! DC DATA NOT VALID -> CA-SLAVE-PLUGIN NOT CONNECTED"); | ||||
|             return false; | ||||
|         } | ||||
|         Utils::printCriticalErrorMsg("DC DATA NOT VALID -> CA-SLAVE-PLUGIN NOT CONNECTED"); | ||||
|         m_hw->dc_autoRequest(true); | ||||
|         QThread::msleep(500); | ||||
|     } | ||||
|  | ||||
|     bool res = false; | ||||
|     bool dcDownloadPossible = true; | ||||
|  | ||||
|     QList<QString>::const_iterator it; | ||||
|     for (it = filesToWorkOn.cbegin(); it != filesToWorkOn.cend(); ++it) { | ||||
|         m_worker->startProgressLoop(); | ||||
|         QString const &fToWorkOn = QDir::cleanPath(m_customerRepository + QDir::separator() + it->trimmed()); | ||||
|         if (fToWorkOn.endsWith("/dc2c.bin") && dcDownloadPossible) { | ||||
| #if 0 | ||||
|             // download for dc possible only once | ||||
|             // download of device-controller should always be the last step | ||||
|             dcDownloadPossible = false; | ||||
|  | ||||
|             if (!m_hw->dcDownloadRequest(fToWorkOn)) {  // initiate download process | ||||
|                 qCritical() << "DOWNLOAD-REQUEST-ERROR FOR" << fToWorkOn; | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             QThread::sleep(2); | ||||
|  | ||||
|             int tries = 5; | ||||
|             while (!m_hw->dcDownloadRunning()) { // may take some time | ||||
|                 if (--tries < 0) { | ||||
|                     qCritical() << QDateTime::currentDateTime().toString(Qt::ISODate) | ||||
|                                 << "(" << __func__ << ":" << __LINE__ << ") DOWNLOAD NOT RUNNING"; | ||||
|                     break; | ||||
|                 } | ||||
|                 QThread::sleep(1); | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             qCritical() << QDateTime::currentDateTime().toString(Qt::ISODate) | ||||
|                         << "(" << __func__ << ":" << __LINE__ << ") DOWNLOAD RUNNING"; | ||||
|  | ||||
|             QThread::sleep(2); | ||||
|  | ||||
|             tries = 5; | ||||
|             while (!m_hw->dcDownloadReportThreadStart()) { // may take some time | ||||
|                 if (--tries < 0) { | ||||
|                     qCritical() << QDateTime::currentDateTime().toString(Qt::ISODate) | ||||
|                                 << "(" << __func__ << ":" << __LINE__ << ") REPORT THREAD NOT STARTED"; | ||||
|                     break; | ||||
|                 } | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             qCritical() << QDateTime::currentDateTime().toString(Qt::ISODate) | ||||
|                         << "(" << __func__ << ":" << __LINE__ << ") REPORT THREAD STARTED"; | ||||
|  | ||||
|             QThread::sleep(2); | ||||
|  | ||||
|             tries = 5; | ||||
|             while (!m_hw->dcDownloadReportRunning()) {     // may take some time | ||||
|                 if (--tries < 0) { | ||||
|                     qCritical() << QDateTime::currentDateTime().toString(Qt::ISODate) | ||||
|                                 << "(" << __func__ << ":" << __LINE__ << ") DOWNLOAD REPORT NOT RUNNING"; | ||||
|                     break; | ||||
|                 } | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             qCritical() << QDateTime::currentDateTime().toString(Qt::ISODate) | ||||
|                         << "(" << __func__ << ":" << __LINE__ << ") DOWNLOAD REPORT RUNNING"; | ||||
|  | ||||
|             tries = 1200; | ||||
|             while (m_hw->dcDownloadReportRunning()) { | ||||
|                 QThread::msleep(1000); | ||||
|                 if (--tries < 0) { | ||||
|                     qCritical() << QDateTime::currentDateTime().toString(Qt::ISODate) | ||||
|                                 << "(" << __func__ << ":" << __LINE__ | ||||
|                                 << ") DOWNLOAD REPORT STILL RUNNING AFTER 20mins"; | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
| #endif | ||||
|             bool updateBinaryRes = true; | ||||
|  | ||||
|             // CONSOLE() | ||||
| #if 0 | ||||
|             m_hw->dc_autoRequest(false);// default: turn auto-request setting off | ||||
|             QThread::sleep(1);          // wait to be sure that there are no more | ||||
|                                         // commands sent to dc-hardware | ||||
|  | ||||
|             if ((updateBinaryRes = updateBinary(fToWorkOn)) == true) { | ||||
|  | ||||
|                 // qCritical() << "downloaded binary" << fToWorkOn; | ||||
|  | ||||
|                 ++displayIndex; | ||||
|                 emit m_worker->appendText(QString("\n(") + QString("%1").arg(displayIndex).rightJustified(2, ' ') + QString(")") | ||||
|                     + QString(" Update ") + QFileInfo(fToWorkOn).fileName(), | ||||
|                     Worker::UPDATE_STEP_DONE); | ||||
|             } | ||||
|  | ||||
|             m_hw->dc_autoRequest(true); // turn auto-request setting on | ||||
|  | ||||
|             // qInfo() << "SET AUTO-REQUEST=TRUE"; | ||||
|  | ||||
|             QStringList const &versions = Update::getDcSoftAndHardWareVersion(); | ||||
|             if (versions.size() >= 2) { | ||||
|                 if (updateBinaryRes == true) { | ||||
|                     qInfo() << "dc-hardware-version (UPDATED)" << versions[0]; | ||||
|                     qInfo() << "dc-firmware-version (UPDATED)" << versions[1]; | ||||
|                 } else { | ||||
|                     qInfo() << "dc-hardware-version (NOT UPDATED)" << versions[0]; | ||||
|                     qInfo() << "dc-firmware-version (NOT UPDATED)" << versions[1]; | ||||
|                 } | ||||
|             } | ||||
| #endif | ||||
|             res = updateBinaryRes; | ||||
|  | ||||
|         } else if (fToWorkOn.contains("DC2C_print", Qt::CaseInsensitive) | ||||
|                 && fToWorkOn.endsWith(".json", Qt::CaseInsensitive)) { | ||||
|             res = true; | ||||
|             int i = fToWorkOn.indexOf("DC2C_print", Qt::CaseInsensitive); | ||||
|             int const templateIdx = fToWorkOn.mid(i).midRef(10, 2).toInt(); | ||||
|             if ((templateIdx < 1) || (templateIdx > 32)) { | ||||
|                 qCritical() << "WRONG TEMPLATE INDEX" << templateIdx; | ||||
|                 res = false; | ||||
|             } else { | ||||
|                 if ((res = updatePrinterTemplate(templateIdx, fToWorkOn))) { | ||||
|                     Utils::printInfoMsg( | ||||
|                         QString("DOWNLOADED PRINTER TEMPLATE %1 WITH INDEX=%2") | ||||
|                             .arg(fToWorkOn) | ||||
|                             .arg(templateIdx)); | ||||
|                     ++displayIndex; | ||||
|                     emit m_worker->appendText(QString("\n(") + QString("%1").arg(displayIndex).rightJustified(3, ' ') + QString(")") | ||||
|                           + QString(" Update ") + QFileInfo(fToWorkOn).fileName(), | ||||
|                             Worker::UPDATE_STEP_DONE); | ||||
|                 } | ||||
|             } | ||||
|         } else if (fToWorkOn.contains("DC2C_cash", Qt::CaseInsensitive) | ||||
|                 && fToWorkOn.endsWith(".json", Qt::CaseInsensitive)) { | ||||
|             res = true; | ||||
|             if ((res = updateCashConf(fToWorkOn))) { | ||||
|                 Utils::printInfoMsg(QString("DOWNLOADED CASH TEMPLATE %1").arg(fToWorkOn)); | ||||
|                 ++displayIndex; | ||||
|                 emit m_worker->appendText(QString("\n(") + QString("%1").arg(displayIndex).rightJustified(3, ' ') + QString(")") | ||||
|                       + QString(" Update ") + QFileInfo(fToWorkOn).fileName(), | ||||
|                         Worker::UPDATE_STEP_DONE); | ||||
|             } | ||||
|         } else if (fToWorkOn.contains("DC2C_conf", Qt::CaseInsensitive) | ||||
|                 && fToWorkOn.endsWith(".json", Qt::CaseInsensitive)) { | ||||
|             res = true; | ||||
|             if ((res= updateConfig(fToWorkOn))) { | ||||
|                 Utils::printInfoMsg(QString("DOWNLOADED CONFIG TEMPLATE %1").arg(fToWorkOn)); | ||||
|                 ++displayIndex; | ||||
|                 emit m_worker->appendText(QString("\n(") + QString("%1").arg(displayIndex).rightJustified(3, ' ') + QString(")") | ||||
|                       + QString(" Update ") + QFileInfo(fToWorkOn).fileName(), | ||||
|                         Worker::UPDATE_STEP_DONE); | ||||
|             } | ||||
|         } else if (fToWorkOn.contains("DC2C_device", Qt::CaseInsensitive) | ||||
|                 && fToWorkOn.endsWith(".json", Qt::CaseInsensitive)) { | ||||
|             res = true; | ||||
|             if ((res = updateDeviceConf(fToWorkOn))) { | ||||
|                 Utils::printInfoMsg(QString("DOWNLOADED DEVICE TEMPLATE %1").arg(fToWorkOn)); | ||||
|                 ++displayIndex; | ||||
|                 emit m_worker->appendText(QString("\n(") + QString("%1").arg(displayIndex).rightJustified(3, ' ') + QString(")") | ||||
|                       + QString(" Update ") + QFileInfo(fToWorkOn).fileName(), | ||||
|                         Worker::UPDATE_STEP_DONE); | ||||
|             } | ||||
|         } else { | ||||
|             qCritical() << "UNKNOWN JSON FILE NAME" << fToWorkOn; | ||||
|             res = false; | ||||
|         } | ||||
|         // m_worker->stopProgressLoop(); | ||||
|         // m_worker->setProgress(100); | ||||
|  | ||||
|         if (res == false) { | ||||
|             break; | ||||
|         } | ||||
|     } // for (it = openLines.cbegin(); it != openLines.end(); ++it) { | ||||
|  | ||||
|     m_hw->dc_autoRequest(true);  // ALWAYS turn autoRequest ON | ||||
|     qDebug() << "SET AUTO-REQUEST=TRUE"; | ||||
|  | ||||
|     return res; | ||||
| } | ||||
| @@ -9,7 +9,7 @@ | ||||
| #include <QProcess> | ||||
| #include <QPluginLoader> | ||||
| 
 | ||||
| #include "plugins/interfaces.h" | ||||
| #include <DeviceController/interfaces.h> | ||||
| 
 | ||||
| #ifdef PTU5 | ||||
| #define SERIAL_PORT "ttymxc2" | ||||
| @@ -32,6 +32,7 @@ class Update : public QObject { | ||||
|     QString m_workingDir; | ||||
|     bool m_maintenanceMode; | ||||
|     bool m_dryRun; | ||||
|     bool m_sys_areDCdataValid; | ||||
| 
 | ||||
|     static QPluginLoader pluginLoader; | ||||
| 
 | ||||
| @@ -57,6 +58,7 @@ public: | ||||
|                     char const *baudrate = "115200"); | ||||
|     virtual ~Update() override; | ||||
|     bool doUpdate(int &displayIndex, QStringList const &linesToWorkOn); | ||||
|     bool checkDownloadedJsonVersions(QStringList const& jsonFileNames); | ||||
| 
 | ||||
|     hwinf *hw() { return m_hw; } | ||||
|     hwinf const *hw() const { return m_hw; } | ||||
| @@ -72,34 +74,28 @@ public: | ||||
| 
 | ||||
| private: | ||||
|     static QString jsonType(enum FileTypeJson type); | ||||
| 
 | ||||
|     DownloadResult sendStatus(int ret) const; | ||||
|     DownloadResult sendNextAddress(int bNum) const; | ||||
|     DownloadResult sendNextDataBlock(QByteArray const &b, int bNum) const; | ||||
|     DownloadResult dc_downloadBinary(QByteArray const &binary) const; | ||||
| 
 | ||||
|     bool startBootloader() const; | ||||
|     bool stopBootloader() const; | ||||
|     bool openSerial(int br, QString baudrate, QString comPort) const; | ||||
|     void closeSerial() const; | ||||
|     bool isSerialOpen() const; | ||||
|     bool resetDeviceController() const; | ||||
|     QByteArray loadBinaryDCFile(QString filename) const; | ||||
|     bool downloadBinaryToDC(QString const &bFile) const; | ||||
|     bool updateDC(QString bFile) const; | ||||
|     bool updatePrinterTemplate(int templateIdx, QString fname) const; | ||||
|     bool updateBinary(char const *fileToSendToDC); | ||||
|     bool updateBinary(QString const &fileToSendToDC); | ||||
|     bool updateConfig(QString jsFileToSendToDC); | ||||
|     bool updateCashConf(QString jsFileToSendToDC); | ||||
|     bool updateDeviceConf(QString jsFileToSendToDC); | ||||
|     bool downloadJson(enum FileTypeJson type, int templateIdx, | ||||
|                       QString jsFileToSendToDC) const; | ||||
|     QStringList getDcSoftAndHardWareVersion(); | ||||
|     QString getFileVersion(QString const& jsonFile); | ||||
| 
 | ||||
| private slots: | ||||
|     void readyReadStandardOutput(); | ||||
|     void readyReadStandardError(); | ||||
|     void finished(int exitCode, QProcess::ExitStatus exitStatus); | ||||
| 
 | ||||
|     void onReportDCDownloadStatus(QString const &status); | ||||
|     void onReportDCDownloadSuccess(QString const &msg); | ||||
|     void onReportDCDownloadFailure(QString const &errorMsg); | ||||
| }; | ||||
| #endif // UPDATE_H_INCLUDED
 | ||||
| @@ -30,6 +30,85 @@ int Utils::read1stLineOfFile(QString fileName) { | ||||
|     return -1; | ||||
| } | ||||
| 
 | ||||
| QString Utils::getLocation(QString fileName) { | ||||
|     QString location("N/A"); | ||||
|     QFile f(fileName); | ||||
|     if (f.exists()) { | ||||
|         if (f.open(QIODevice::ReadOnly | QIODevice::Text)) { | ||||
|             QTextStream in(&f); | ||||
|             in.setCodec("UTF-8"); | ||||
|             while(!in.atEnd()) { | ||||
|                 QString const &line = in.readLine(); | ||||
|                 if (line.indexOf("Project", Qt::CaseInsensitive) != -1) { | ||||
|                     int const c = line.indexOf(":"); | ||||
|                     if (c != -1) { | ||||
|                         location = line.mid(c+1); | ||||
|                         if (!location.isEmpty()) { | ||||
|                             location = location.replace(QChar(','), QString("")); | ||||
|                             return location.replace(QChar('"'), QString("")).trimmed(); | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     return location; | ||||
| } | ||||
| 
 | ||||
| QString Utils::getTariffVersion(QString fileName) { | ||||
|     QString version("N/A"); | ||||
|     QFile f(fileName); | ||||
|     if (f.exists()) { | ||||
|         if (f.open(QIODevice::ReadOnly | QIODevice::Text)) { | ||||
|             QTextStream in(&f); | ||||
|             in.setCodec("UTF-8"); | ||||
|             while(!in.atEnd()) { | ||||
|                 QString const &line = in.readLine(); | ||||
|                 if (line.indexOf("Version", Qt::CaseInsensitive) != -1) { | ||||
|                     int const c = line.indexOf(":"); | ||||
|                     if (c != -1) { | ||||
|                         version = line.mid(c+1); | ||||
|                         if (!version.isEmpty()) { | ||||
|                             version = version.replace(QChar(','), QString("")); | ||||
|                             return version.replace(QChar('"'), QString("")).trimmed(); | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     return version; | ||||
| } | ||||
| 
 | ||||
| QString Utils::getTariffInfo(QString fileName) { | ||||
|     QString info("N/A"); | ||||
|     QFile f(fileName); | ||||
|     if (f.exists()) { | ||||
|         if (f.open(QIODevice::ReadOnly | QIODevice::Text)) { | ||||
|             QTextStream in(&f); | ||||
|             in.setCodec("UTF-8"); | ||||
|             while(!in.atEnd()) { | ||||
|                 QString const &line = in.readLine(); | ||||
|                 if (line.indexOf("Info", Qt::CaseInsensitive) != -1) { | ||||
|                     int const c = line.indexOf(":"); | ||||
|                     if (c != -1) { | ||||
|                         info = line.mid(c+1); | ||||
|                         if (!info.isEmpty()) { | ||||
|                             info = info.replace(QChar(','), QString("")); | ||||
|                             return info.replace(QChar('"'), QString("")).trimmed(); | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     return info; | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| QString Utils::zoneName(quint8 i) { | ||||
|     static constexpr char const *zName[] = { | ||||
|         "", | ||||
| @@ -201,21 +280,24 @@ bool Utils::sameFilesInDirs(QDir const &dir1, QDir const &dir2, | ||||
|         fileNameLst2 << i2.next().fileName(); | ||||
|     } | ||||
| 
 | ||||
|     QString dirPath1 = dir1.absolutePath(); | ||||
|     QString dirPath2 = dir2.absolutePath(); | ||||
| 
 | ||||
|     if (fileNameLst1.isEmpty()) { | ||||
|         qCritical() << "DIR1" << dir1.dirName() << " DOES NOT CONTAIN EXPECTED FILES"; | ||||
|         qCritical() << "DIR1" << dirPath1 << " DOES NOT CONTAIN EXPECTED FILES"; | ||||
|         return false; | ||||
|     } | ||||
|     if (fileNameLst2.isEmpty())  { | ||||
|         qCritical() << "DIR1" << dir2.dirName() << " DOES NOT CONTAIN EXPECTED FILES"; | ||||
|         qCritical() << "DIR1" << dirPath2 << " DOES NOT CONTAIN EXPECTED FILES"; | ||||
|         return false; | ||||
|     } | ||||
|     if (fileNameLst1 != fileNameLst2) { | ||||
|         printCriticalErrorMsg(dir1.dirName() + " AND " + dir2.dirName() | ||||
|         printCriticalErrorMsg(dirPath1 + " AND " + dirPath2 | ||||
|                             + " DIFFER: [" + fileNameLst1.join(',') + "],[" | ||||
|                             + fileNameLst2.join(',') + "]"); | ||||
|         return false; | ||||
|     } else { | ||||
|         printInfoMsg(dir1.dirName() + " AND " + dir2.dirName() | ||||
|         printInfoMsg(dirPath1 + " AND " + dirPath2 | ||||
|                     + " ARE EQUAL: [" + fileNameLst1.join(',') + "]"); | ||||
|     } | ||||
| 
 | ||||
| @@ -231,21 +313,21 @@ bool Utils::sameFilesInDirs(QDir const &dir1, QDir const &dir2, | ||||
|     } | ||||
| 
 | ||||
|     if (gitBlobLst1.isEmpty()) { | ||||
|         qCritical() << "DIR1" << dir1.dirName() << " DOES NOT CONTAIN EXPECTED FILES"; | ||||
|         qCritical() << "DIR1" << dirPath1 << " DOES NOT CONTAIN EXPECTED FILES"; | ||||
|         return false; | ||||
|     } | ||||
|     if (gitBlobLst2.isEmpty())  { | ||||
|         qCritical() << "DIR1" << dir2.dirName() << " DOES NOT CONTAIN EXPECTED FILES"; | ||||
|         qCritical() << "DIR1" << dirPath2 << " DOES NOT CONTAIN EXPECTED FILES"; | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     if (gitBlobLst1 != gitBlobLst2) { | ||||
|         printCriticalErrorMsg(dir1.dirName() + " AND " + dir2.dirName() | ||||
|         printCriticalErrorMsg(dirPath1 + " AND " + dirPath2 | ||||
|                             + " DIFFER: [" + gitBlobLst1.join(',') + "],[" | ||||
|                             + gitBlobLst2.join(',') + "]"); | ||||
|         return false; | ||||
|     } else { | ||||
|         printInfoMsg(dir1.dirName() + " AND " + dir2.dirName() | ||||
|         printInfoMsg(dirPath1 + " AND " + dirPath2 | ||||
|                     + " CONTAIN SAME GIT-BLOBS FOR FILES: [" + fileNameLst1.join(',') + "]"); | ||||
| 
 | ||||
|     } | ||||
| @@ -12,6 +12,9 @@ | ||||
| 
 | ||||
| namespace Utils { | ||||
|     int read1stLineOfFile(QString fileName); | ||||
|     QString getLocation(QString fileName); | ||||
|     QString getTariffVersion(QString fileName); | ||||
|     QString getTariffInfo(QString fileName); | ||||
|     QString zoneName(quint8 i); | ||||
|     void printCriticalErrorMsg(QString const &errorMsg, bool upper=false, bool lower=false); | ||||
|     void printCriticalErrorMsg(QStringList const &errorMsg); | ||||
| @@ -17,9 +17,10 @@ | ||||
| #include <QJsonParseError> | ||||
| #include <Qt> | ||||
| #include <QScopedPointer> | ||||
| #include <QRegularExpression> | ||||
| 
 | ||||
| #include "message_handler.h" | ||||
| #include "plugins/interfaces.h" | ||||
| #include <DeviceController/interfaces.h> | ||||
| #include "ismas/ismas_client.h" | ||||
| #include "progress_event.h" | ||||
| #include "mainwindow.h" | ||||
| @@ -56,6 +57,9 @@ const QMap<UPDATE_STEP, const char*> Worker::smap ( | ||||
|         INSERT_ELEMENT(UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::INITIAL_CLONE_WITHOUT_ACTIVE_ISMAS_TRIGGER), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::INITIAL_CLONE_WITH_ACTIVE_ISMAS_TRIGGER), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::PULL_NEW_BRANCH), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::PULL_NEW_BRANCH_FAILURE), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::PULL_NEW_BRANCH_SUCCESS), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::CHECKOUT_BRANCH), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::CHECKOUT_BRANCH_SUCCESS), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::CHECKOUT_BRANCH_FAILURE), | ||||
| @@ -96,12 +100,15 @@ const QMap<UPDATE_STEP, const char*> Worker::smap ( | ||||
|         INSERT_ELEMENT(UPDATE_STEP::SAVE_LOGS_SUCCESS), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::SAVE_LOGS_FAILURE), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::SEND_LAST_VERSION), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::UPDATE_FINALIZE), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::UPDATE_SUCCEEDED), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::UPDATE_NOT_NECESSARY), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::UPDATE_FAILED), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::UPDATE_ACTIVATED), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::FINISHED), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::DEBUG), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::ERROR) | ||||
|         INSERT_ELEMENT(UPDATE_STEP::ERROR), | ||||
|         INSERT_ELEMENT(UPDATE_STEP::NONE) | ||||
| #undef INSERT_ELEMENT | ||||
| }); | ||||
| 
 | ||||
| @@ -116,6 +123,8 @@ Worker::Worker(int customerNr, | ||||
|                QString pluginName, | ||||
|                QString workingDirectory, | ||||
|                bool noUpdatePsaHardware, | ||||
|                bool alwaysDownloadConfig, | ||||
|                bool alwaysDownloadDC, | ||||
|                bool dryRun, | ||||
|                QObject *parent, | ||||
|                char const *serialInterface, | ||||
| @@ -131,11 +140,14 @@ Worker::Worker(int customerNr, | ||||
|   , m_customerRepositoryPath(QString("%1/%2.git").arg(repositoryUrl).arg(m_customerNrStr)) | ||||
|   , m_customerRepository(QDir::cleanPath(m_workingDirectory + QDir::separator() + m_customerNrStr)) | ||||
|   , m_noUpdatePsaHardware(noUpdatePsaHardware) | ||||
|   , m_alwaysDownloadConfig(alwaysDownloadConfig) | ||||
|   , m_alwaysDownloadDC(alwaysDownloadDC) | ||||
|   , m_dryRun(dryRun) | ||||
|   , m_parent(parent) | ||||
|   , m_serialInterface(serialInterface) | ||||
|   , m_baudrate(baudrate) | ||||
|   , m_gc(m_customerNrStr, m_customerRepository, m_workingDirectory, m_branchName, this) | ||||
|   , m_versionInfo(QStringList()) | ||||
|   , m_osVersion(getOsVersion()) | ||||
|   , m_atbqtVersion(getATBQTVersion()) | ||||
|   , m_atbUpdateToolVersion(getATBUpdateToolVersion()) | ||||
| @@ -157,6 +169,17 @@ Worker::Worker(int customerNr, | ||||
| 
 | ||||
|     // TODO: turn object into singleton
 | ||||
|     instance = this; | ||||
|     m_lastFailedUpdateStep = UPDATE_STEP::NONE; | ||||
| 
 | ||||
|     if (m_noUpdatePsaHardware == false) { | ||||
|         m_update = new Update(this, | ||||
|                               QDir::cleanPath(m_workingDirectory + QDir::separator() + m_customerNrStr), | ||||
|                               m_customerNrStr, | ||||
|                               m_branchName, | ||||
|                               m_pluginDir, | ||||
|                               m_pluginName, | ||||
|                               m_workingDirectory); | ||||
|     } | ||||
| 
 | ||||
|     this->setObjectName("worker-object"); | ||||
|     QDir::setCurrent(m_workingDirectory); | ||||
| @@ -165,6 +188,10 @@ Worker::Worker(int customerNr, | ||||
| } | ||||
| 
 | ||||
| Worker::~Worker() { | ||||
|     if (m_update != nullptr) { | ||||
|         delete m_update; | ||||
|         m_update = nullptr; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| void Worker::displayProgressInMainWindow(int progress) { | ||||
| @@ -210,6 +237,12 @@ bool Worker::isRepositoryCorrupted() { | ||||
|             Utils::printCriticalErrorMsg("CORRUPTED CUSTOMER REPOSITORY .GIT DOES NOT EXIST"); | ||||
|             return true; | ||||
|         } | ||||
|         QDir customerRepositoryEtc(QDir::cleanPath(m_customerRepository + QDir::separator() + "etc/")); | ||||
|         if (!customerRepositoryEtc.exists()) { | ||||
|             // should never happen
 | ||||
|             Utils::printCriticalErrorMsg(QString("CORRUPTED CUSTOMER REPOSITORY %1/etc DOES NOT EXIST").arg(m_customerRepository)); | ||||
|             return true; | ||||
|         } | ||||
|     } | ||||
|     return false; | ||||
| } | ||||
| @@ -244,15 +277,72 @@ void Worker::privateUpdate() { | ||||
|     GUI() << (ISMAS() << (CONSOLE() << UPDATE_STEP::STARTED)); | ||||
| 
 | ||||
|     QScopedPointer<UpdateProcessRunning> upr(new UpdateProcessRunning(this)); | ||||
|     QStringList lst; | ||||
| 
 | ||||
|     ////////////////////////////////////////////////////////////////////////////
 | ||||
|     //
 | ||||
|     //                         CHECK UPDATE TRIGGER
 | ||||
|     //
 | ||||
|     ////////////////////////////////////////////////////////////////////////////
 | ||||
|     // NOTE: make sure that nothing is sent to ISMAS during updateTriggerSet
 | ||||
|     ISMAS() << UPDATE_STEP::CHECK_ISMAS_TRIGGER; | ||||
|     m_ismasTriggerActive = false; | ||||
|     m_updateNotNecessary = false; | ||||
| 
 | ||||
|     if (QDir(m_customerRepository).exists()) { // ignore a possibly corrupted repository
 | ||||
|         m_ismasTriggerActive = updateTriggerSet(); | ||||
|         if (m_ismasTriggerActive == false) { | ||||
|             QDateTime const ¤t = QDateTime::currentDateTime(); | ||||
|             m_automaticUpdate = (current.time().hour() < 4); | ||||
|             m_versionInfo = m_gc.gitShowReason(m_branchName); | ||||
| 
 | ||||
|             qCritical() << "***"; | ||||
|             qCritical() << "privateUpdate ............. m_versionInfo:" << m_versionInfo; | ||||
|             qCritical() << "privateUpdate ......... m_automaticUpdate:" << m_automaticUpdate; | ||||
| 
 | ||||
|             if (m_automaticUpdate) { // update has been triggered within [00:00:00, 00:03:59]
 | ||||
|                 m_updateNotNecessary = true; | ||||
|                 m_ismasTriggerStatusMessage = QStringList(QString("NO UPDATE NECESSARY (%1)").arg(current.toString(Qt::ISODate))); | ||||
| 
 | ||||
|                 qCritical() << "privateUpdate m_ismasTriggerStatusMessage:" | ||||
|                             << QStringList(m_ismasTriggerStatusMessage); | ||||
|                 qCritical() << "***"; | ||||
| 
 | ||||
|                 // the customer-repository does exist, and the ISMAS-trigger is
 | ||||
|                 // *NOT* "WAIT", but from 00:00:00 - 00:03:59 this counts as an
 | ||||
|                 // automatic update
 | ||||
| 
 | ||||
|                 QStringList lst = m_ismasTriggerStatusMessage; | ||||
|                 // trigger message to ISMAS and CONSOLE
 | ||||
|                 ISMAS(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_SUCCESS)); | ||||
|                 // overwrite m_lastFailedUpdateStep
 | ||||
|                 m_lastFailedUpdateStep = UPDATE_STEP::NONE; | ||||
|                 return; | ||||
|             } | ||||
| 
 | ||||
|             qCritical() << "***"; | ||||
| 
 | ||||
|             // the customer-repository does exist, but the ISMAS-trigger is
 | ||||
|             // *NOT* "WAIT", so STOP the update procedure
 | ||||
|             return; | ||||
|         } | ||||
|         // the customer-repository does exist, and the ISMAS-trigger is "WAIT",
 | ||||
|         // so continue the update procedure
 | ||||
|     } else { | ||||
|         // the customer-repository does not exist, so PROCEED with the
 | ||||
|         // update procedure, even if ISMAS-trigger is not correctly set ("WAIT")
 | ||||
|     } | ||||
| 
 | ||||
|     emit this->disableExit(); | ||||
| 
 | ||||
|     QDir customerRepository(m_customerRepository); | ||||
|     QDir customerRepositoryEtc(QDir::cleanPath(m_customerRepository + QDir::separator() + "etc/")); | ||||
| 
 | ||||
|     CONSOLE() << UPDATE_STEP::CHECK_SANITY; | ||||
|     CONSOLE() << (ISMAS() << UPDATE_STEP::CHECK_SANITY); | ||||
| 
 | ||||
|     m_clone = false; | ||||
|     m_repairClone = false; | ||||
|     m_initialClone = false; | ||||
|     m_pulledNewBranch = false; | ||||
|                                     // the customer repository is cloned or
 | ||||
|                                     // repaired/re-cloned without checking the
 | ||||
|                                     // ISMAS-trigger (WAIT-)button.
 | ||||
| @@ -271,26 +361,27 @@ void Worker::privateUpdate() { | ||||
| 
 | ||||
|     if (isRepositoryCorrupted()) {  // a not-existing repository is not meant
 | ||||
|                                     // to be corrupted
 | ||||
|         CONSOLE() << UPDATE_STEP::CHECK_SANITY_FAILURE; | ||||
|         CONSOLE() << (ISMAS() << UPDATE_STEP::CHECK_SANITY_FAILURE); | ||||
|         if ((continueUpdate = repairCorruptedRepository()) == true) { | ||||
|             m_repairClone = true; | ||||
|             CONSOLE() << UPDATE_STEP::REPOSITORY_RECOVERED_SUCCESS; | ||||
|             CONSOLE() << (ISMAS() << UPDATE_STEP::REPOSITORY_RECOVERED_SUCCESS); | ||||
|         } else { | ||||
|             ISMAS() << (GUI() << (CONSOLE() << UPDATE_STEP::REPOSITORY_RECOVERED_FAILURE)); | ||||
|             return; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     CONSOLE() << UPDATE_STEP::CHECK_SANITY_SUCCESS; | ||||
|     CONSOLE() << (ISMAS() << UPDATE_STEP::CHECK_SANITY_SUCCESS); | ||||
| 
 | ||||
|     if (continueUpdate) { | ||||
|         if ((continueUpdate = customerRepository.exists()) == false) { | ||||
|             m_initialClone = (m_repairClone == false); | ||||
|             GUI() << (CONSOLE() << UPDATE_STEP::CLONE_REPOSITORY); | ||||
|             ISMAS() << (GUI() << (CONSOLE() << UPDATE_STEP::CLONE_REPOSITORY)); | ||||
|             for (int i = 0; i < 5; ++i) {   // try to checkout git repository
 | ||||
|                 setProgress(i);             // and switch to branch
 | ||||
|                 if (m_gc.gitCloneAndCheckoutBranch()) { | ||||
|                     if (!isRepositoryCorrupted()) { | ||||
|                         m_versionInfo = m_gc.gitShowReason(m_branchName); | ||||
|                         GUI() << (ISMAS() << (CONSOLE() << UPDATE_STEP::CLONE_REPOSITORY_SUCCESS)); | ||||
|                         continueUpdate = true; | ||||
|                         m_clone = true; | ||||
| @@ -301,7 +392,7 @@ void Worker::privateUpdate() { | ||||
|             } | ||||
| 
 | ||||
|             if (continueUpdate == false) { | ||||
|                 GUI() << (ISMAS() << (CONSOLE() << UPDATE_STEP::CLONE_REPOSITORY_FAILURE)); | ||||
|                 GUI() << (ISMAS() << (CONSOLE() << (m_lastFailedUpdateStep = UPDATE_STEP::CLONE_REPOSITORY_FAILURE))); | ||||
|                 return; | ||||
|             } | ||||
| 
 | ||||
| @@ -315,36 +406,37 @@ void Worker::privateUpdate() { | ||||
| 
 | ||||
|             CONSOLE() << UPDATE_STEP::CHECK_REPOSITORY; | ||||
|             if (isRepositoryCorrupted()) { | ||||
|                 ISMAS() << (GUI() << (CONSOLE() << UPDATE_STEP::CHECK_REPOSITORY_FAILURE)); | ||||
|                 ISMAS() << (GUI() << (CONSOLE() << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_REPOSITORY_FAILURE))); | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     m_versionInfo = m_gc.gitShowReason(m_branchName); | ||||
| 
 | ||||
|     CONSOLE() << UPDATE_STEP::CHECK_REPOSITORY_SUCCESS; | ||||
|     setProgress(_CHECKOUT_REPOSITORY_SUCCESS); | ||||
| 
 | ||||
| 
 | ||||
|     ////////////////////////////////////////////////////////////////////////////
 | ||||
|     //
 | ||||
|     //                         CHECK UPDATE TRIGGER
 | ||||
|     //
 | ||||
|     ////////////////////////////////////////////////////////////////////////////
 | ||||
|     m_ismasTriggerActive = false; | ||||
|     if ((continueUpdate = updateTriggerSet()) == false) { | ||||
|     if (m_clone == false) { | ||||
|         if (m_ismasTriggerActive == false) { | ||||
|             return; | ||||
|         } else { | ||||
|             GUI() << (ISMAS() << (CONSOLE() << UPDATE_STEP::CHECK_ISMAS_TRIGGER_SUCCESS)); | ||||
|             setProgress(_CHECK_ISMAS_TRIGGER_SUCCESS); | ||||
|         } | ||||
|     } else { | ||||
|         if (m_initialClone) { | ||||
|             GUI() << (ISMAS() << (CONSOLE() << UPDATE_STEP::INITIAL_CLONE_WITHOUT_ACTIVE_ISMAS_TRIGGER)); | ||||
|         } | ||||
|         return; | ||||
|     } else { | ||||
|         GUI() << (ISMAS() << (CONSOLE() << UPDATE_STEP::INITIAL_CLONE_WITH_ACTIVE_ISMAS_TRIGGER)); | ||||
|     } | ||||
| 
 | ||||
|     if (m_ismasTriggerActive == false) {// make it explicit again: only if the
 | ||||
|                                         // ismas trigger is active ('WAIT'),
 | ||||
|                                         // then proceed
 | ||||
|         return; | ||||
|     } | ||||
|     GUI() << (ISMAS() << (CONSOLE() << UPDATE_STEP::CHECK_ISMAS_TRIGGER_SUCCESS)); | ||||
|     setProgress(_CHECK_ISMAS_TRIGGER_SUCCESS); | ||||
| 
 | ||||
|         if (m_clone == false) {  // if it is an (initial) clone, then
 | ||||
|             return;              // run the whole update process:
 | ||||
|         }                        // sync tariff-files, download jsons,
 | ||||
|     }                            // download device controller
 | ||||
| 
 | ||||
| 
 | ||||
|     ////////////////////////////////////////////////////////////////////////////
 | ||||
| @@ -355,7 +447,9 @@ void Worker::privateUpdate() { | ||||
|     if ((continueUpdate = customerEnvironment()) == false) { | ||||
|         return; | ||||
|     } | ||||
|     CONSOLE() << UPDATE_STEP::CHECKOUT_BRANCH_SUCCESS; | ||||
|     m_versionInfo = m_gc.gitShowReason(m_branchName); | ||||
|     lst = QStringList(QString(smap[UPDATE_STEP::CHECKOUT_BRANCH_SUCCESS])); | ||||
|     ISMAS(lst) << (CONSOLE(lst) << UPDATE_STEP::CHECKOUT_BRANCH); | ||||
|     setProgress(_CHECKOUT_BRANCH_SUCCESS); | ||||
| 
 | ||||
| 
 | ||||
| @@ -367,9 +461,23 @@ void Worker::privateUpdate() { | ||||
|     if ((continueUpdate = filesToUpdate()) == false) { | ||||
|         return; | ||||
|     } | ||||
|     ISMAS() << (GUI() << (CONSOLE() << UPDATE_STEP::UPDATE_REPOSITORY_SUCCESS)); | ||||
|     m_versionInfo = m_gc.gitShowReason(m_branchName); | ||||
|     lst = QStringList(QString(smap[UPDATE_STEP::UPDATE_REPOSITORY_SUCCESS])); | ||||
|     ISMAS() << (GUI() << (CONSOLE() << UPDATE_STEP::UPDATE_REPOSITORY)); | ||||
|     setProgress(_UPDATE_REPOSITORY_SUCCESS); | ||||
| 
 | ||||
|     ////////////////////////////////////////////////////////////////////////////
 | ||||
|     //
 | ||||
|     //             (R)SYNC THE REPOSITORY WITH THE LOCAL FILEYSTEM
 | ||||
|     //
 | ||||
|     ////////////////////////////////////////////////////////////////////////////
 | ||||
|     if ((continueUpdate = syncCustomerRepositoryAndFS()) == false) { | ||||
|         return; | ||||
|     } | ||||
|     lst = QStringList(QString(smap[UPDATE_STEP::SYNC_CUSTOMER_REPOSITORY_SUCCESS])); | ||||
|     ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::SYNC_CUSTOMER_REPOSITORY_SUCCESS)); | ||||
|     setProgress(_SYNC_CUSTOMER_REPOSITORY_SUCCESS); | ||||
| 
 | ||||
| 
 | ||||
|     ////////////////////////////////////////////////////////////////////////////
 | ||||
|     //
 | ||||
| @@ -379,7 +487,8 @@ void Worker::privateUpdate() { | ||||
|     if ((continueUpdate = execOpkgCommands()) == false) { | ||||
|         return; | ||||
|     } | ||||
|     GUI() << (CONSOLE() << UPDATE_STEP::EXEC_OPKG_COMMAND_SUCCESS); | ||||
|     lst = QStringList(QString(smap[UPDATE_STEP::EXEC_OPKG_COMMAND_SUCCESS])); | ||||
|     GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::EXEC_OPKG_COMMANDS); | ||||
|     setProgress(_EXEC_OPKG_COMMAND_SUCCESS); | ||||
| 
 | ||||
| 
 | ||||
| @@ -391,22 +500,10 @@ void Worker::privateUpdate() { | ||||
|     if ((continueUpdate = downloadFilesToPSAHardware()) == false) { | ||||
|         return; | ||||
|     } | ||||
|     GUI() << (CONSOLE() << UPDATE_STEP::DOWNLOAD_FILES_TO_PSA_HARDWARE_SUCCESS); | ||||
|     lst = QStringList(QString(smap[UPDATE_STEP::DOWNLOAD_FILES_TO_PSA_HARDWARE_SUCCESS])); | ||||
|     ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::DOWNLOAD_FILES_TO_PSA_HARDWARE)); | ||||
|     setProgress(_DOWNLOAD_FILES_TO_PSA_HARDWARE_SUCCESS); | ||||
| 
 | ||||
| 
 | ||||
|     ////////////////////////////////////////////////////////////////////////////
 | ||||
|     //
 | ||||
|     //             (R)SYNC THE REPOSITORY WITH THE LOCAL FILEYSTEM
 | ||||
|     //
 | ||||
|     ////////////////////////////////////////////////////////////////////////////
 | ||||
|     if ((continueUpdate = syncCustomerRepositoryAndFS()) == false) { | ||||
|         return; | ||||
|     } | ||||
|     ISMAS() << (GUI() << (CONSOLE() << UPDATE_STEP::SYNC_CUSTOMER_REPOSITORY_SUCCESS)); | ||||
|     setProgress(_SYNC_CUSTOMER_REPOSITORY_SUCCESS); | ||||
| 
 | ||||
| 
 | ||||
|     ////////////////////////////////////////////////////////////////////////////
 | ||||
|     //
 | ||||
|     //                            FUTURE: SAVE LOG FILES
 | ||||
| @@ -418,28 +515,8 @@ void Worker::privateUpdate() { | ||||
|     // ISMAS() << (GUI() << (CONSOLE() << UPDATE_STEP::SAVE_LOGS_SUCCESS));
 | ||||
|     setProgress(_SAVE_LOGS_SUCCESS); | ||||
| 
 | ||||
| 
 | ||||
|     ////////////////////////////////////////////////////////////////////////////
 | ||||
|     //
 | ||||
|     //                      FINAL MESSAGES (PART 1)
 | ||||
|     //
 | ||||
|     ////////////////////////////////////////////////////////////////////////////
 | ||||
|     ISMAS() << (GUI() << (CONSOLE() << UPDATE_STEP::UPDATE_SUCCEEDED)); | ||||
|     setProgress(_UPDATE_SUCCEEDED); | ||||
| 
 | ||||
|     ISMAS() << (GUI() << (CONSOLE() << UPDATE_STEP::UPDATE_ACTIVATED)); | ||||
|     setProgress(_UPDATE_ACTIVATED); | ||||
| 
 | ||||
|     ISMAS() << (GUI() << (CONSOLE() << UPDATE_STEP::FINISHED)); | ||||
|     setProgress(_FINISHED); | ||||
| 
 | ||||
| 
 | ||||
|     ////////////////////////////////////////////////////////////////////////////
 | ||||
|     //
 | ||||
|     //                 FINAL MESSAGES (PART 2): SEND-LAST-VERSION
 | ||||
|     //                 (destructor of struct UpdateProcessRunning)
 | ||||
|     //
 | ||||
|     ////////////////////////////////////////////////////////////////////////////
 | ||||
|     // final messages: see destructor of UpdateProcessRunning subclass
 | ||||
|     m_lastFailedUpdateStep = UPDATE_STEP::NONE; | ||||
| } | ||||
| 
 | ||||
| bool Worker::updateTriggerSet() { | ||||
| @@ -447,31 +524,38 @@ bool Worker::updateTriggerSet() { | ||||
|     // (WAIT-button) is activated even in case of initial checkout
 | ||||
|     static const QString func = "UPDATE-TRIGGER-SET"; | ||||
| 
 | ||||
|     if (m_withoutIsmasDirectPort) { // useful for testing
 | ||||
|         return true; | ||||
|     } | ||||
|     // if (m_withoutIsmasDirectPort) { // useful for testing
 | ||||
|     //    return true;
 | ||||
|     //}
 | ||||
| 
 | ||||
|     GUI() << (ISMAS() << (CONSOLE() << UPDATE_STEP::CHECK_ISMAS_TRIGGER)); | ||||
|     m_ismasTriggerStatusMessage.clear(); | ||||
| 
 | ||||
|     QString triggerValue(""); | ||||
|     GUI() << (CONSOLE() << UPDATE_STEP::CHECK_ISMAS_TRIGGER); | ||||
| 
 | ||||
|     for (int repeat = 1; repeat <= 100; ++repeat) { | ||||
|     bool const automaticUpdate = (QDateTime::currentDateTime().time().hour() < 4); | ||||
| 
 | ||||
|     QString triggerValue("NOT CHECKED YET"); | ||||
|     static constexpr int const repeats = 15; | ||||
|     for (int repeat = 1; repeat <= repeats; ++repeat) { | ||||
| 
 | ||||
|         if (repeat > 1) { | ||||
|             int const startMs = QTime::currentTime().msecsSinceStartOfDay(); | ||||
|             int const durationMs = QTime::currentTime().msecsSinceStartOfDay() - startMs; | ||||
|             QString const &s = QString("elapsed: %1.%2s").arg(durationMs / 1000).arg(durationMs % 1000); | ||||
|             CONSOLE(QStringList(func) << s) << UPDATE_STEP::DEBUG; | ||||
|             QStringList lst = (m_ismasTriggerStatusMessage = (QStringList(func) << s)); | ||||
|             CONSOLE(lst) << UPDATE_STEP::DEBUG; | ||||
|         } else { | ||||
|             CONSOLE(QStringList(func) << QString("-> REPEAT=%1").arg(repeat)) << UPDATE_STEP::DEBUG; | ||||
|             QStringList lst = (m_ismasTriggerStatusMessage = (QStringList(func) << QString("-> REPEAT=%1 (%2)").arg(repeat).arg(repeats-repeat))); | ||||
|             CONSOLE(lst) << UPDATE_STEP::DEBUG; | ||||
|         } | ||||
| 
 | ||||
|         if ((repeat % 10) == 0) { | ||||
|         if ((repeat % 8) == 0) { | ||||
|             CONSOLE(QStringList(func) << "RESTART APISM") << UPDATE_STEP::DEBUG; | ||||
|             Command c("systemctl restart apism"); | ||||
|             if (c.execute("/tmp")) { | ||||
|                 QThread::sleep(20); // give APISM some time to reconnect
 | ||||
|                 CONSOLE(QStringList(func) << "RESTART APISM DONE") << UPDATE_STEP::DEBUG; | ||||
|                 QStringList lst = (m_ismasTriggerStatusMessage = (QStringList(func) << "RESTART APISM DONE")); | ||||
|                 CONSOLE(lst) << UPDATE_STEP::DEBUG; | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
| @@ -480,31 +564,35 @@ bool Worker::updateTriggerSet() { | ||||
|                     IsmasClient::APISM::DIRECT_PORT, "#M=APISM#C=REQ_ISMASPARAMETER#J={}")) { | ||||
| 
 | ||||
|             QString const &msg = QString("APISM RESPONSE(%1)=(").arg(repeat) + result.value() + ")"; | ||||
|             CONSOLE(QStringList(func) << msg) << UPDATE_STEP::DEBUG; | ||||
|             QStringList lst = (m_ismasTriggerStatusMessage = (QStringList(func) << msg)); | ||||
|             CONSOLE(lst) << UPDATE_STEP::DEBUG; | ||||
| 
 | ||||
|             QJsonParseError parseError; | ||||
|             QJsonDocument document(QJsonDocument::fromJson(result.value().toUtf8(), &parseError)); | ||||
|             if (parseError.error != QJsonParseError::NoError) { | ||||
|                 QStringList lst(QString("INVALID JSON MSG: PARSING FAILED (json=<START>%1<END> error=[%2] str=[%3] offset=[%4])") | ||||
|                 m_ismasTriggerStatusMessage = QStringList(QString("INVALID JSON MSG: PARSING FAILED (json=<START>%1<END> error=[%2] str=[%3] offset=[%4])") | ||||
|                                                 .arg(msg) | ||||
|                                                 .arg(parseError.error) | ||||
|                                                 .arg(parseError.errorString()) | ||||
|                                                 .arg(parseError.offset)); | ||||
|                 GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE); | ||||
|                 break; | ||||
|                 QStringList lst = m_ismasTriggerStatusMessage; | ||||
|                 ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE))); | ||||
|                 return false; | ||||
|             } | ||||
|             if (!document.isObject()) { | ||||
|                 QStringList lst(QString("not a json-object %1").arg(result.value())); | ||||
|                 GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE); | ||||
|                 break; | ||||
|                 m_ismasTriggerStatusMessage = QStringList(QString("not a json-object %1").arg(result.value())); | ||||
|                 QStringList lst = m_ismasTriggerStatusMessage; | ||||
|                 ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE))); | ||||
|                 return false; | ||||
|             } | ||||
| 
 | ||||
|             QJsonObject obj = document.object(); | ||||
| 
 | ||||
|             // always look for an 'error' first
 | ||||
|             if (obj.contains("error")) { | ||||
|                 QStringList lst(obj.value("error").toString()); | ||||
|                 CONSOLE(QStringList(lst)) << UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE; | ||||
|                 m_ismasTriggerStatusMessage = QStringList(obj.value("error").toString()); | ||||
|                 QStringList lst = m_ismasTriggerStatusMessage; | ||||
|                 CONSOLE(QStringList(lst)) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE); | ||||
|                 QThread::sleep(6); | ||||
|                 continue; | ||||
|             } | ||||
| @@ -519,31 +607,35 @@ bool Worker::updateTriggerSet() { | ||||
|                         int const customerNr = obj.value("Custom_ID").toInt(-1); | ||||
|                         int const machineNr = obj.value("Device_ID").toInt(-1); | ||||
|                         if (customerNr != m_customerNr) { | ||||
|                             QStringList lst(QString("CUSTOMER-NR (%1) != LOCAL CUSTOMER-NR (%2)") | ||||
|                             m_ismasTriggerStatusMessage = QStringList(QString("CUSTOMER-NR (%1) != LOCAL CUSTOMER-NR (%2)") | ||||
|                                                             .arg(customerNr).arg(m_customerNr)); | ||||
|                             GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE); | ||||
|                             break; | ||||
|                             QStringList lst = m_ismasTriggerStatusMessage; | ||||
|                             ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE))); | ||||
|                             return false; | ||||
|                         } | ||||
|                         if (machineNr != m_machineNr) { | ||||
|                             QStringList lst(QString("MACHINE-NR (%1) != LOCAL MACHINE-NR (%2)") | ||||
|                             m_ismasTriggerStatusMessage = QStringList(QString("MACHINE-NR (%1) != LOCAL MACHINE-NR (%2)") | ||||
|                                                             .arg(machineNr).arg(m_machineNr)); | ||||
|                             GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE); | ||||
|                             break; | ||||
|                             QStringList lst = m_ismasTriggerStatusMessage; | ||||
|                             ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE))); | ||||
|                             return false; | ||||
|                         } | ||||
|                     } else { | ||||
|                         QStringList lst("Dev_ID DOES NOT CONTAIN Custom_ID AND/OR Device_ID"); | ||||
|                         GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE); | ||||
|                         break; | ||||
|                         QStringList lst(QString("Dev_ID DOES NOT CONTAIN Custom_ID AND/OR Device_ID (LINE=%1)").arg(__LINE__)); | ||||
|                         ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE))); | ||||
|                         return false; | ||||
|                     } | ||||
|                 } else { | ||||
|                     QStringList lst("Dev_ID KEY NOT A JSON-OBJECT"); | ||||
|                     GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE); | ||||
|                     break; | ||||
|                     m_ismasTriggerStatusMessage = QStringList(QString("Dev_ID KEY NOT A JSON-OBJECT (LINE=%1)").arg(__LINE__)); | ||||
|                     QStringList lst = m_ismasTriggerStatusMessage; | ||||
|                     ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE))); | ||||
|                     return false; | ||||
|                 } | ||||
|             } else { | ||||
|                 QStringList lst("Dev_ID KEY NOT AVAILABLE"); | ||||
|                 GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE); | ||||
|                 break; | ||||
|                 m_ismasTriggerStatusMessage = QStringList(QString("Dev_ID KEY NOT AVAILABLE (LINE=%1)").arg(__LINE__)); | ||||
|                 QStringList lst = m_ismasTriggerStatusMessage; | ||||
|                 ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE))); | ||||
|                 return false; | ||||
|             } | ||||
| 
 | ||||
|             if (obj.contains("Fileupload")) { | ||||
| @@ -552,56 +644,86 @@ bool Worker::updateTriggerSet() { | ||||
|                     obj = v.toObject(); | ||||
|                     if (obj.contains("TRG")) { | ||||
|                         if ((triggerValue = obj.value("TRG").toString()) == "WAIT") { | ||||
|                             m_ismasTriggerStatusMessage = QStringList("ISMAS_UPDATE-TRIGGER SET TO WAIT"); | ||||
|                             m_ismasTriggerActive = true; | ||||
|                             return m_ismasTriggerActive; | ||||
|                         } else | ||||
|                         if (QRegExp("\\s*").exactMatch(triggerValue)) { // check for whitespace
 | ||||
|                             QStringList lst("empty update trigger"); | ||||
|                             m_ismasTriggerStatusMessage = QStringList(QString("%1 EMPTY UPDATE TRIGGER (%2)").arg(repeat).arg(repeats-repeat)); | ||||
|                             QStringList lst = m_ismasTriggerStatusMessage; | ||||
|                             if (m_clone) { | ||||
|                                 GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::CHECK_ISMAS_TRIGGER_WRONG_VALUE); | ||||
|                                 ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_WRONG_VALUE))); | ||||
|                                                     // if the customer repository has just been cloned
 | ||||
|                                 break;              // it is OK the ISMAS trigger might not be 'WAIT'
 | ||||
|                             } else { | ||||
|                                 GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE); | ||||
|                                 return false;       // it is OK the ISMAS trigger might not be 'WAIT'
 | ||||
|                             } | ||||
|                             // not a clone and empty update-trigger
 | ||||
|                             if (automaticUpdate) { | ||||
|                                 // do not inform ISMAS in case of automatic update, because the
 | ||||
|                                 // update is not necessary as the trigger-button is not set to WAIT.
 | ||||
|                                 GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_WRONG_VALUE)); | ||||
|                                 return false; | ||||
|                             } | ||||
| 
 | ||||
|                             CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE); | ||||
| 
 | ||||
|                             QThread::sleep(6); | ||||
|                             continue; | ||||
|                         } else { | ||||
|                             // if the download-button once has the wrong value, it will never recover
 | ||||
|                             // if the download-button once has a wrong value, it will never recover
 | ||||
|                             if (m_clone) { | ||||
|                                 GUI() << (CONSOLE() << UPDATE_STEP::CHECK_ISMAS_TRIGGER_WRONG_VALUE); | ||||
|                                 m_ismasTriggerStatusMessage = QStringList(QString("TRIGGER-VALUE='%1' != 'WAIT'").arg(triggerValue)); | ||||
|                                 QStringList lst = m_ismasTriggerStatusMessage; | ||||
|                                 if (automaticUpdate) { | ||||
|                                     // do not inform ISMAS in case of automatic update
 | ||||
|                                     GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_WRONG_VALUE)); | ||||
|                                 } else { | ||||
|                                 QStringList lst(QString("TRIGGER-VALUE=<%1> NOT 'WAIT'").arg(triggerValue)); | ||||
|                                 GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE); | ||||
|                             } | ||||
|                             break; | ||||
|                                     ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_WRONG_VALUE))); | ||||
|                                 } | ||||
|                             } else { | ||||
|                         QStringList lst("TRG key not available"); | ||||
|                         GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE); | ||||
|                         break; | ||||
|                                 m_ismasTriggerStatusMessage = QStringList(QString("TRIGGER-VALUE='%1' != 'WAIT'").arg(triggerValue)); | ||||
|                                 QStringList lst = m_ismasTriggerStatusMessage; | ||||
|                                 if (automaticUpdate) { | ||||
|                                     // do not inform ISMAS in case of automatic update
 | ||||
|                                     GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_WRONG_VALUE)); | ||||
|                                 } else { | ||||
|                                     ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE))); | ||||
|                                 } | ||||
|                             } | ||||
|                             return false; | ||||
|                         } | ||||
|                     } else { | ||||
|                     QStringList lst("Fileupload not a json-object"); | ||||
|                     GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE); | ||||
|                     break; | ||||
|                         m_ismasTriggerStatusMessage = QStringList(QString("TRG key not available (LINE=%1)").arg(__LINE__)); | ||||
|                         QStringList lst = m_ismasTriggerStatusMessage; | ||||
|                         ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE))); | ||||
|                         return false; | ||||
|                     } | ||||
|                 } else { | ||||
|                 QStringList lst(QString("Fileupload not available")); | ||||
|                 GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE); | ||||
|                 break; | ||||
|                     m_ismasTriggerStatusMessage = QStringList(QString("Fileupload not a json-object (LINE=%1)").arg(__LINE__)); | ||||
|                     QStringList lst = m_ismasTriggerStatusMessage; | ||||
|                     ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE))); | ||||
|                     return false; | ||||
|                 } | ||||
|             } else { | ||||
|             QStringList lst = QStringList(QString("no ISMAS response")); | ||||
|             GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE); | ||||
|                 m_ismasTriggerStatusMessage = QStringList(QString("Fileupload not available (LINE=%1)").arg(__LINE__)); | ||||
|                 QStringList lst = m_ismasTriggerStatusMessage; | ||||
|                 ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE))); | ||||
|                 return false; | ||||
|             } | ||||
|         } else { | ||||
|             m_ismasTriggerStatusMessage = QStringList(QString("no ISMAS response (LINE=%1)").arg(__LINE__)); | ||||
|             QStringList lst = m_ismasTriggerStatusMessage; | ||||
|             CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE); | ||||
|             QThread::sleep(6); | ||||
|             continue; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     if (m_initialClone == false) { | ||||
|         if (!triggerValue.contains("WAIT", Qt::CaseInsensitive)) { | ||||
|             QStringList lst(QString("ISMAS_UPDATE-TRIGGER-NOT-SET-OR-WRONG: VALUE=(") + triggerValue + ")"); | ||||
|             ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE)); | ||||
|             m_ismasTriggerStatusMessage = QStringList(QString("ISMAS_UPDATE-TRIGGER-NOT-SET-OR-WRONG: VALUE=(") + triggerValue + ")"); | ||||
|             QStringList lst = m_ismasTriggerStatusMessage; | ||||
|             ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECK_ISMAS_TRIGGER_FAILURE))); | ||||
|             return false; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
| @@ -615,15 +737,43 @@ bool Worker::customerEnvironment() { | ||||
|     ISMAS() << (GUI() << (CONSOLE() << UPDATE_STEP::CHECKOUT_BRANCH)); | ||||
| 
 | ||||
|     if (QDir(m_customerRepository).exists()) { | ||||
|         if (m_clone == false) { | ||||
|             if (m_gc.branchExistsRemotely()) { | ||||
|                 QString msg; | ||||
|                 QStringList lst; | ||||
|                 if (!m_gc.branchExistsLocally()) { | ||||
|                     lst.clear(); | ||||
|                     msg = QString("PULLING OF NEW BRANCH " + m_branchName + " DOES NOT EXIST LOCALLY"); | ||||
|                     lst << msg; | ||||
|                     CONSOLE(lst) << UPDATE_STEP::PULL_NEW_BRANCH; | ||||
|                     if (!m_gc.gitPullNewBranches()) { | ||||
|                         lst.clear(); | ||||
|                         msg = QString("PULLING OF NEW BRANCH " + m_branchName + " FAILED"); | ||||
|                         lst << msg; | ||||
|                         ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::PULL_NEW_BRANCH_FAILURE))); | ||||
|                         return false; | ||||
|                     } else { | ||||
|                         lst.clear(); | ||||
|                         msg = QString("PULLING OF NEW BRANCH " + m_branchName + " SUCCESS"); | ||||
|                         lst << msg; | ||||
|                         ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::PULL_NEW_BRANCH_SUCCESS))); | ||||
|                         m_pulledNewBranch = true; | ||||
|                     } | ||||
|                 } else { | ||||
|                     m_pulledNewBranch = false; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         if (m_gc.gitCheckoutBranch()) { | ||||
|             return true; | ||||
|         } else { | ||||
|             QStringList lst(QString("CHECKOUT OF " + m_customerRepository + "FAILED")); | ||||
|             ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::CHECKOUT_BRANCH_FAILURE)); | ||||
|             ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECKOUT_BRANCH_FAILURE))); | ||||
|         } | ||||
|     } else {// cannot happen
 | ||||
|         QStringList lst(QString(m_customerRepository + " DOES NOT EXIST")); | ||||
|         ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << UPDATE_STEP::CHECKOUT_BRANCH_FAILURE)); | ||||
|         ISMAS(lst) << (GUI(lst) << (CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::CHECKOUT_BRANCH_FAILURE))); | ||||
|     } | ||||
| 
 | ||||
|     return false; | ||||
| @@ -638,6 +788,33 @@ bool Worker::filesToUpdate() { | ||||
|     // always execute contents of opkg_commands-file
 | ||||
|     m_filesToUpdate << "etc/psa_update/opkg_commands"; | ||||
| 
 | ||||
|     if ((m_clone || m_pulledNewBranch) && m_alwaysDownloadConfig) { | ||||
|         // always download all json-config files, even if none of them have been
 | ||||
|         // changed in the git repository. useful for first installation.
 | ||||
|         QDir dir(QDir::cleanPath(m_customerRepository + QDir::separator() + "etc/psa_config")); | ||||
|         if (dir.exists()) { | ||||
|             QStringList jsons = dir.entryList(QStringList() << "DC2C*.json", QDir::Files); | ||||
|             if (!jsons.isEmpty()) { | ||||
|                 for (QStringList::size_type i=0; i<jsons.size(); ++i) { | ||||
|                     m_filesToUpdate << QDir::cleanPath(QString("etc/psa_config/") + jsons.at(i)); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     if ((m_clone || m_pulledNewBranch) && m_alwaysDownloadConfig) { | ||||
|         // always download the last dc-binary, even if not changed in the
 | ||||
|         // git repository. useful for first installation.
 | ||||
|         QDir dir(QDir::cleanPath(m_customerRepository + QDir::separator() + "etc/dc")); | ||||
|         if (dir.exists()) { | ||||
|             QStringList dc = dir.entryList(QStringList() << "dc2c.bin", QDir::Files, | ||||
|                                            QDir::SortFlag::Time | QDir::SortFlag::Reversed); | ||||
|             if (!dc.isEmpty()) { | ||||
|                 m_filesToUpdate << QDir::cleanPath(QString("etc/dc/") + dc.first()); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     if (std::optional<QString> changes = m_gc.gitPull()) { | ||||
|         if (!changes.value().contains("Already up to date")) { | ||||
|             if (std::optional<QStringList> changedFileNames = m_gc.gitDiff(changes.value())) { | ||||
| @@ -645,14 +822,16 @@ bool Worker::filesToUpdate() { | ||||
|             } | ||||
|         } | ||||
|         m_filesToUpdate.removeDuplicates(); | ||||
| 
 | ||||
|         qCritical() << "(" << __func__ << ":" << __LINE__ << ") FILES-TO-UPDATE" << m_filesToUpdate; | ||||
| 
 | ||||
|         GUI(m_filesToUpdate) << (CONSOLE(m_filesToUpdate) << UPDATE_STEP::FILES_TO_UPDATE); | ||||
|         setProgress(_FILES_TO_UPDATE); | ||||
|     } else { | ||||
|         ISMAS() << (GUI() << (CONSOLE() << UPDATE_STEP::UPDATE_REPOSITORY_FAILURE)); | ||||
|         ISMAS() << (GUI() << (CONSOLE() << (m_lastFailedUpdateStep = UPDATE_STEP::UPDATE_REPOSITORY_FAILURE))); | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
|     return true; | ||||
| } | ||||
| 
 | ||||
| @@ -666,8 +845,7 @@ bool Worker::computeFilesToDownload() { | ||||
|             fName.contains("DC2C_cash", Qt::CaseInsensitive)) { | ||||
|             m_filesToDownload << fName;  // download printer-config-files
 | ||||
|         } else { | ||||
|             static const QRegularExpression version("^.*dc2c[.][0-9]{1,2}[.][0-9]{1,2}[.]bin.*$"); | ||||
|             if (fName.contains(version)) { | ||||
|             if (fName.contains("dc2c.bin")) { | ||||
|                 m_filesToDownload << fName; // download device controller
 | ||||
|             } | ||||
|         } | ||||
| @@ -688,7 +866,7 @@ bool Worker::execOpkgCommands() { | ||||
|                     if (f.open(QIODevice::ReadOnly)) { | ||||
|                         QTextStream in(&f); | ||||
|                         m_opkgCommands.clear(); | ||||
|                         bool executeOpkgCommandFailed = false; | ||||
|                         QStringList opkgErrorLst; | ||||
|                         while (!in.atEnd()) { | ||||
|                             QString line = in.readLine(); | ||||
|                             static const QRegularExpression comment("^\\s*#.*$"); | ||||
| @@ -696,7 +874,7 @@ bool Worker::execOpkgCommands() { | ||||
|                                 // found opkg command
 | ||||
|                                 QString opkgCommand = line.trimmed(); | ||||
|                                 if (!executeOpkgCommand(opkgCommand)) { | ||||
|                                     executeOpkgCommandFailed = true; | ||||
|                                     opkgErrorLst << opkgCommand; | ||||
|                                 } else { | ||||
|                                     QString cmd = "\n  " + opkgCommand; | ||||
|                                     emit appendText(cmd); | ||||
| @@ -750,14 +928,15 @@ bool Worker::execOpkgCommands() { | ||||
|                             } | ||||
|                         } | ||||
|                         f.close(); | ||||
|                         if (!executeOpkgCommandFailed) { | ||||
|                         if (opkgErrorLst.size() == 0) { | ||||
|                             if (m_opkgCommands.size() > 0) { | ||||
|                                 m_displayIndex = 1; | ||||
|                                 GUI() << UPDATE_STEP::EXEC_OPKG_COMMAND_SUCCESS; | ||||
|                                 ISMAS() << (GUI() << (CONSOLE() << UPDATE_STEP::EXEC_OPKG_COMMAND_SUCCESS)); | ||||
|                                 setProgress(_EXEC_OPKG_COMMAND_SUCCESS); | ||||
|                             } | ||||
|                         } else { | ||||
|                             m_displayIndex = 1; | ||||
|                             ISMAS(opkgErrorLst) << (GUI(opkgErrorLst) << (CONSOLE() << (m_lastFailedUpdateStep = UPDATE_STEP::EXEC_OPKG_COMMAND_FAILURE))); | ||||
|                             GUI() << UPDATE_STEP::EXEC_OPKG_COMMAND_FAILURE; | ||||
|                             setProgress(_EXEC_OPKG_COMMAND_FAILURE); | ||||
|                             return false; | ||||
| @@ -773,25 +952,28 @@ bool Worker::execOpkgCommands() { | ||||
| bool Worker::downloadFilesToPSAHardware() { | ||||
|     m_displayIndex = 0; | ||||
| 
 | ||||
|     GUI(m_filesToUpdate) << (CONSOLE(m_filesToUpdate) << UPDATE_STEP::DOWNLOAD_FILES_TO_PSA_HARDWARE); | ||||
|     GUI(m_filesToDownload) << (CONSOLE(m_filesToDownload) << UPDATE_STEP::DOWNLOAD_FILES_TO_PSA_HARDWARE); | ||||
|     setProgress(_DOWNLOAD_FILES_TO_PSA_HARDWARE); | ||||
| 
 | ||||
|     if (m_noUpdatePsaHardware == false) { | ||||
|         if (computeFilesToDownload()) { | ||||
|             CONSOLE(m_filesToDownload) << UPDATE_STEP::FILES_TO_DOWNLOAD; | ||||
|         if (computeFilesToDownload() > 0) { | ||||
|             QStringList lst = m_filesToDownload; | ||||
|             ISMAS(lst) << (CONSOLE(lst) << UPDATE_STEP::FILES_TO_DOWNLOAD); | ||||
| 
 | ||||
|             Update update(this, | ||||
|                           QDir::cleanPath(m_workingDirectory + QDir::separator() + m_customerNrStr), | ||||
|                           m_customerNrStr, | ||||
|                           m_branchName, | ||||
|                           m_pluginDir, | ||||
|                           m_pluginName, | ||||
|                           m_workingDirectory); | ||||
|             if (m_update && m_update->doUpdate(m_displayIndex, m_filesToDownload)) { | ||||
|                 // prepared for use: at the moment, the dc-library does not work
 | ||||
|                 // as expected.
 | ||||
| 
 | ||||
|             return update.doUpdate(m_displayIndex, m_filesToDownload); | ||||
|         } else { | ||||
|             CONSOLE(QStringList("NO FILES TO DOWNLOAD TO PSA-HW")) << UPDATE_STEP::DOWNLOAD_FILES_TO_PSA_HARDWARE_FAILURE; | ||||
|                 // static const QRegularExpression re("^.*\\.json$");
 | ||||
|                 // return update.checkDownloadedJsonVersions(m_filesToDownload.filter(re));
 | ||||
| 
 | ||||
|                 return true; | ||||
|             } | ||||
| 
 | ||||
|             CONSOLE(lst) << (m_lastFailedUpdateStep = UPDATE_STEP::DOWNLOAD_FILES_TO_PSA_HARDWARE_FAILURE); | ||||
|             setProgress(_DOWNLOAD_FILES_TO_PSA_HARDWARE_FAILURE); | ||||
| 
 | ||||
|             return false; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
| @@ -808,7 +990,7 @@ bool Worker::syncCustomerRepositoryAndFS() { | ||||
|                             QStringList() << "-c" << "mkdir -p /etc/psa_config /etc/dc /etc/psa_tariff")) { | ||||
|                 qCritical() << "COULD NOT EXECUTE '" << md.command() << "' exitCode=(" << md.exitCode() << ")"; | ||||
|             } | ||||
|             QString const params("-vvv                    " | ||||
|             QString const params("-v                      " | ||||
|                                  "--recursive             " | ||||
|                                  "--progress              " | ||||
|                                  "--checksum              " | ||||
| @@ -871,14 +1053,16 @@ bool Worker::syncCustomerRepositoryAndFS() { | ||||
|                 QDir dir1(QDir::cleanPath(m_customerRepository + QDir::separator() + "etc/psa_tariff")); | ||||
|                 QDir dir2("/etc/psa_tariff"); | ||||
|                 if (Utils::sameFilesInDirs(dir1, dir2)) { | ||||
|                     ISMAS() << (GUI() << (CONSOLE() << UPDATE_STEP::SYNC_CUSTOMER_REPOSITORY_SUCCESS)); | ||||
|                     CONSOLE() <<  QDir::cleanPath(m_customerRepository + QDir::separator() + "etc/psa_tariff") | ||||
|                               << "AND /etc/psa_tariff ARE DIFFERENT: CHANGED CUSTOMER-NUMBER?"; | ||||
|                 } | ||||
|                 CONSOLE() << UPDATE_STEP::SYNC_CUSTOMER_REPOSITORY_SUCCESS; | ||||
|                 setProgress(_SYNC_CUSTOMER_REPOSITORY_SUCCESS); | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     } | ||||
|     ISMAS() << (GUI() << (CONSOLE() << UPDATE_STEP::SYNC_CUSTOMER_REPOSITORY_FAILURE)); | ||||
|     ISMAS() << (GUI() << (CONSOLE() << (m_lastFailedUpdateStep = UPDATE_STEP::SYNC_CUSTOMER_REPOSITORY_FAILURE))); | ||||
|     setProgress(_SYNC_CUSTOMER_REPOSITORY_FAILURE); | ||||
|     return false; | ||||
| } | ||||
| @@ -911,7 +1095,7 @@ QString Worker::getATBUpdateToolYoctoVersion() { | ||||
| 
 | ||||
| QString Worker::getAPISMYoctoVersion() { | ||||
|     if (QFile::exists("/var/lib/opkg/status")) { | ||||
|         QString const cmd = QString("echo -n $(cat /var/lib/opkg/status | grep -A1 apism | tail -n 1 | cut -d':' -f2 | cut -d' ' -f2)"); | ||||
|         QString const cmd = QString("echo -n $(cat /var/lib/opkg/status | grep -A1 -e apism[[:space:]]*$ | tail -n 1 | cut -d':' -f2 | cut -d' ' -f2)"); | ||||
|         Command c("bash"); | ||||
|         if (c.execute("/tmp", QStringList() << "-c" << cmd)) { | ||||
|             return c.getCommandResult(); // 1.4.1.0-r4
 | ||||
| @@ -942,6 +1126,17 @@ QString Worker::getAPISMYoctoInstallationStatus()  { | ||||
|     return "N/A"; | ||||
| } | ||||
| 
 | ||||
| QString Worker::getDCVersionPreparedForDownload(QString const &filename) { | ||||
|     if (QFile::exists(filename)) {  // <customer-repo/etc/dc/dc2c.bin>
 | ||||
|         QString const cmd = QString("strings %1 | grep -e DC2[Cc]\\. | head -n1").arg(filename); | ||||
|         Command c("bash"); | ||||
|         if (c.execute("/tmp", QStringList() << "-c" << cmd)) { | ||||
|             return c.getCommandResult(); // DC2c.04.42  14.09.2023
 | ||||
|         } | ||||
|     } | ||||
|     return "N/A"; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| QString Worker::getATBQTVersion() const { | ||||
|     QString const cmd = QString("echo -n $(/opt/app/ATBAPP/ATBQT -v | head -n 2 | cut -d':' -f2)"); | ||||
| @@ -1059,16 +1254,52 @@ PSAInstalled Worker::getPSAInstalled() { | ||||
|     QString absPathName; | ||||
|     QString absPathNameRepository; | ||||
| 
 | ||||
|     psaInstalled.versionInfo.lastCommit = ""; | ||||
|     psaInstalled.versionInfo.reason = ""; | ||||
|     psaInstalled.versionInfo.created = ""; | ||||
|     psaInstalled.versionInfo.lastCommit = "N/A"; | ||||
|     psaInstalled.versionInfo.reason = "N/A"; | ||||
|     psaInstalled.versionInfo.created = "N/A"; | ||||
| 
 | ||||
|     QStringList versionInfo = m_gc.gitShowReason(m_branchName); | ||||
|     if (versionInfo.size() == 3) { | ||||
|         psaInstalled.versionInfo.lastCommit = versionInfo.at(0); | ||||
|         psaInstalled.versionInfo.reason = versionInfo.at(1); | ||||
|         psaInstalled.versionInfo.created = versionInfo.at(2); | ||||
|     if (m_versionInfo.size() == 3) { | ||||
|         QString const &lastCommit = m_versionInfo.at(0); | ||||
|         QString reason = m_versionInfo.at(1); | ||||
|         QDateTime const dt = QDateTime::fromString(m_versionInfo.at(2), Qt::ISODate); | ||||
|         QString version{""}; | ||||
|         QString date{""}; | ||||
|         if (dt.isValid()) { | ||||
|             date += " "; | ||||
|             date += dt.date().toString(Qt::ISODate); | ||||
|         } | ||||
|         static const QRegularExpression re("^\\s*(\\d+)\\.(\\d+)\\.(\\d+)(.*$)"); | ||||
|         QRegularExpressionMatch match = re.match(reason); | ||||
|         if (match.hasMatch()) { | ||||
|             int const lastCapturedIndex = match.lastCapturedIndex(); | ||||
|             if (lastCapturedIndex >= 1) { | ||||
|                 version += " v"; | ||||
|                 version += match.captured(1);   // major
 | ||||
|             } | ||||
|             if (lastCapturedIndex >= 2) { | ||||
|                 version += "."; | ||||
|                 version += match.captured(2);   // minor
 | ||||
|             } | ||||
|             if (lastCapturedIndex >= 3) { | ||||
|                 version += "."; | ||||
|                 version += match.captured(3);   // patch
 | ||||
|             } | ||||
|             if (lastCapturedIndex >= 4) {       // rest after version
 | ||||
|                 reason = match.captured(4); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         psaInstalled.versionInfo.lastCommit = QString("%1%2").arg(lastCommit).arg(version); | ||||
|         psaInstalled.versionInfo.reason = reason; | ||||
|         psaInstalled.versionInfo.created = m_versionInfo.at(2); | ||||
|     } | ||||
| 
 | ||||
|     //qCritical() << "";
 | ||||
|     //qCritical() << "VERSION-INFO";
 | ||||
|     //qCritical() << "LAST-COMMIT" << psaInstalled.versionInfo.lastCommit;
 | ||||
|     //qCritical() << "REASON" << psaInstalled.versionInfo.reason;
 | ||||
|     //qCritical() << "CREATED" << psaInstalled.versionInfo.created;
 | ||||
|     //qCritical() << "";
 | ||||
| 
 | ||||
|     if (m_zoneNr != 0) { | ||||
|         QString const &n = QString("%1").arg(m_zoneNr).rightJustified(2, '0'); | ||||
| @@ -1080,10 +1311,10 @@ PSAInstalled Worker::getPSAInstalled() { | ||||
|         psaInstalled.tariff.size = getFileSize(absPathName); | ||||
|         psaInstalled.tariff.zone = m_zoneNr; | ||||
|         psaInstalled.tariff.loadTime = Utils::getTariffLoadTime(absPathName); | ||||
|         psaInstalled.tariff.project = Utils::getLocation(absPathName); | ||||
|         psaInstalled.tariff.version = Utils::getTariffVersion(absPathName); | ||||
|         psaInstalled.tariff.info = Utils::getTariffInfo(absPathName); | ||||
|     } | ||||
|     psaInstalled.tariff.project = "Szeged"; | ||||
|     psaInstalled.tariff.info = "N/A"; | ||||
|     psaInstalled.tariff.version = "N/A"; | ||||
| 
 | ||||
|     psaInstalled.hw.linuxVersion = getOsVersion(); | ||||
|     psaInstalled.hw.cpuSerial = m_cpuSerial; | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										55
									
								
								common/include/System.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								common/include/System.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | ||||
| #ifndef SYSTEM_H | ||||
| #define SYSTEM_H | ||||
|  | ||||
|  | ||||
| #include <QObject> | ||||
|  | ||||
| class System : public QObject | ||||
| { | ||||
|     Q_OBJECT | ||||
|  | ||||
| private: | ||||
|     explicit System(QObject *parent = 0); | ||||
|  | ||||
|     static QString errorMsg; | ||||
|  | ||||
|  | ||||
| public: | ||||
|  | ||||
|  | ||||
|     static bool checkForSDCard(); | ||||
|     static bool umountSDCard(); | ||||
|  | ||||
|     static bool checkForUSBStick(); | ||||
|     static QString getUSBMountPath(); | ||||
|     static QString getUSBDeviceName(); | ||||
|     static bool umountUSBStick(); | ||||
|  | ||||
|     static bool test_mount(const QString& dev, const QString &mountpoint); | ||||
|     static bool test_dir(const QString& dir); | ||||
|  | ||||
|     static quint32 getFreeDiskSpace(const QString& dev); | ||||
|     static quint32 getFreeDiskSpaceSDCard(); | ||||
|     static bool isSDCardWritable(); | ||||
|     static quint32 getFreeDiskSpaceUSBStick(); | ||||
|     static bool isUSBStickWritable(); | ||||
|  | ||||
|     static quint8 createLogFileBackup(const QString & targzfile, const QString &filelistfile = ""); | ||||
|     static QString getSDCardMountPath(); | ||||
|     static QString getSDCardDeviceName(); | ||||
|  | ||||
|     static QString getUniqueDeviceId(); | ||||
|  | ||||
|     static QString getErrorMessage(); | ||||
|  | ||||
|     static QString readStringFromFile(const QString & filename); | ||||
|     static int readIntFromFile(const QString & filename); | ||||
|  | ||||
|     static QString getPTU4SerialNumber(); | ||||
|     static QString getPTU4MACAddress(); | ||||
|  | ||||
|  | ||||
|  | ||||
| }; | ||||
|  | ||||
| #endif // SYSTEM_H | ||||
							
								
								
									
										23
									
								
								common/include/message_handler.h
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										23
									
								
								common/include/message_handler.h
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| #ifndef MESSAGE_HANDLER_H_INCLUDED | ||||
| #define MESSAGE_HANDLER_H_INCLUDED | ||||
|  | ||||
| #include <QtGlobal> | ||||
| #ifdef __linux__ | ||||
| #include <syslog.h> | ||||
| #endif | ||||
|  | ||||
| int getDebugLevel(); | ||||
| void setDebugLevel(int newDebugLevel); | ||||
|  | ||||
| bool messageHandlerInstalled(); | ||||
| QtMessageHandler atbInstallMessageHandler(QtMessageHandler handler); | ||||
|  | ||||
| #if QT_VERSION < QT_VERSION_CHECK(5, 0, 0) | ||||
| // typedef void (*QtMessageHandler)(QtMsgType, const char *); | ||||
| void atbDebugOutput(QtMsgType type, const char *msg); | ||||
| #elif QT_VERSION < QT_VERSION_CHECK(6, 0, 0) | ||||
| // typedef void (*QtMessageHandler)(QtMsgType, const QMessageLogContext &, const QString &); | ||||
| void atbDebugOutput(QtMsgType type, const QMessageLogContext &context, const QString &msg); | ||||
| #endif | ||||
|  | ||||
| #endif // MESSAGE_HANDLER_H_INCLUDED | ||||
							
								
								
									
										647
									
								
								common/src/System.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										647
									
								
								common/src/System.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,647 @@ | ||||
| #include "System.h" | ||||
| #include <QFile> | ||||
| #include <QFileInfo> | ||||
| #include <QDir> | ||||
| #include <QTextStream> | ||||
| #include <QProcess> | ||||
| #include <QRegExp> | ||||
| #include <QStringList> | ||||
| #include <QDebug> | ||||
|  | ||||
| //#include "version.h" | ||||
|  | ||||
|  | ||||
| QString System::errorMsg = ""; | ||||
|  | ||||
| System::System(QObject *parent) : | ||||
|     QObject(parent) | ||||
| { | ||||
|  | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| /******************************************************************************** | ||||
|  * static function to return error message. | ||||
|  * | ||||
|  */ | ||||
| QString System::getErrorMessage() | ||||
| { | ||||
|     return errorMsg; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| /******************************************************************************** | ||||
|  * static function to check if a writable SD-card is in the cardslot. | ||||
|  * | ||||
|  * This function only checks a certain path or a mount entry. It does not mount | ||||
|  * a SD-card, this must be provided by the OS (automounter). | ||||
|  * | ||||
|  * returns 1 if a SD-card is available, 0 otherwise. | ||||
|  */ | ||||
| bool System::checkForSDCard() | ||||
| { | ||||
| #if defined (ARCH_DesktopLinux) | ||||
|     // DEBUG / TEST: | ||||
|     return true; | ||||
| #endif | ||||
|  | ||||
|  | ||||
|  | ||||
|     if (!test_mount(getSDCardDeviceName(), getSDCardMountPath())) { | ||||
|         qDebug() << "check for SD-card failed: \n" | ||||
|                  << "    device name is: " << getSDCardDeviceName() | ||||
|                  << "    mount path is:  " << getSDCardMountPath(); | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     return true; | ||||
| } | ||||
|  | ||||
|  | ||||
| /******************************************************************************** | ||||
|  * static function for umount sd-card. | ||||
|  * | ||||
|  */ | ||||
| bool System::umountSDCard() | ||||
| { | ||||
| #if defined (ARCH_DesktopLinux) | ||||
|     // DEBUG / TEST: | ||||
|     return true; | ||||
| #endif | ||||
|  | ||||
|     QProcess process; | ||||
|     process.setProcessChannelMode(QProcess::MergedChannels); | ||||
|  | ||||
|     QString commandString = "umount " + getSDCardMountPath(); | ||||
|  | ||||
|     process.start(commandString); | ||||
|     if (!process.waitForStarted()) { | ||||
|         errorMsg = "System::umountSDCard(): ERROR: waitForStarted()"; | ||||
|         return false; | ||||
|     } | ||||
|     if (!process.waitForFinished(600000)) { | ||||
|         errorMsg = "System::umountSDCard(): ERROR: " + process.errorString(); | ||||
|         qDebug() << errorMsg; | ||||
|         return false; | ||||
|     } | ||||
|     else { | ||||
|         QByteArray bytes = process.readAll(); | ||||
|         QStringList lines = QString(bytes).split("\n"); | ||||
|         foreach (QString line, lines) { | ||||
|             qDebug() << "System::umountSDCard() line:    " << line; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     return true; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| /******************************************************************************** | ||||
|  * static function for umount usb-stick. | ||||
|  * | ||||
|  */ | ||||
| bool System::umountUSBStick() | ||||
| { | ||||
| #if defined (ARCH_DesktopLinux) | ||||
|     // DEBUG / TEST: | ||||
|     return true; | ||||
| #endif | ||||
|  | ||||
|     QProcess process; | ||||
|     process.setProcessChannelMode(QProcess::MergedChannels); | ||||
|  | ||||
|     QString commandString = "umount " + getUSBMountPath(); | ||||
|  | ||||
|     process.start(commandString); | ||||
|     if (!process.waitForStarted()) { | ||||
|         errorMsg = "System::umountUSBStick(): ERROR: waitForStarted()"; | ||||
|         return false; | ||||
|     } | ||||
|     if (!process.waitForFinished(600000)) { | ||||
|         errorMsg = "System::umountUSBStick(): ERROR: " + process.errorString(); | ||||
|         qDebug() << errorMsg; | ||||
|         return false; | ||||
|     } | ||||
|     else { | ||||
|         QByteArray bytes = process.readAll(); | ||||
|         QStringList lines = QString(bytes).split("\n"); | ||||
|         foreach (QString line, lines) { | ||||
|             qDebug() << "System::umountUSBStick() line:    " << line; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     return true; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| /***************************************************************************** | ||||
|  * test, if a certain device is already mounted | ||||
|  * | ||||
|  * e.g. a mounted SD-card on PTU4: | ||||
|  *  - device is '/dev/mmcblk0p1' | ||||
|  *  - mount path is '/media/mmcblk0p1' | ||||
|  * | ||||
|  * e.g. a mounted USB-Stick on PTU4: | ||||
|  *  - device is '/dev/sda' | ||||
|  *  - mount path is '/media/sda' | ||||
|  */ | ||||
| bool System::test_mount(const QString& dev, const QString& mountpoint) | ||||
| { | ||||
|     bool mounted = false; | ||||
|     QProcess process; | ||||
|     process.setProcessChannelMode(QProcess::MergedChannels); | ||||
|  | ||||
|     qDebug() << "System::test_mount() dev = " << dev; | ||||
|     qDebug() << "System::test_mount() mountpoint = " << mountpoint; | ||||
|  | ||||
|     //normalize mountpoint: (mountpoint is without trailing '/') | ||||
|     QString tmp_mountpoint =  QString(mountpoint).remove(QRegExp("/*$")); | ||||
|  | ||||
|     QString commandString = "mount"; | ||||
|  | ||||
|     process.start(commandString); | ||||
|     if (!process.waitForStarted()) { | ||||
|         errorMsg = "System::test_mount(): ERROR: waitForStarted()"; | ||||
|         return false; | ||||
|     } | ||||
|     if (!process.waitForFinished(600000)) { | ||||
|         errorMsg = "System::test_mount(): ERROR: " + process.errorString(); | ||||
|         qDebug() << errorMsg; | ||||
|         return false; | ||||
|     } | ||||
|     else { | ||||
|         QByteArray bytes = process.readAll(); | ||||
|         QStringList lines = QString(bytes).split("\n"); | ||||
|         foreach (QString line, lines) { | ||||
|             //qDebug() << "System::test_mount() line:    " << line; | ||||
|  | ||||
|             //if (line.contains(dev)) qDebug() << "System::test_mount() line contains dev"; | ||||
|             //if (line.contains(tmp_mountpoint)) qDebug() << "System::test_mount() line contains mountpoint"; | ||||
|  | ||||
|             if (line.contains(dev) && line.contains(tmp_mountpoint)) { | ||||
|                 qDebug() << "System::test_mount(): " << dev << " is mounted on " << tmp_mountpoint; | ||||
|                 mounted = true; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     return mounted; | ||||
| } | ||||
|  | ||||
|  | ||||
| /***************************************************************************** | ||||
|  *  test, if a certain directory is existing and readable | ||||
|  */ | ||||
| bool System::test_dir(const QString& dir) | ||||
| { | ||||
|     QFileInfo fileInfo(dir); | ||||
|     if (! fileInfo.isDir() ) {   return false;  } | ||||
|  | ||||
|     return true; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| bool System::checkForUSBStick() | ||||
| { | ||||
| #if defined (ARCH_DesktopLinux) | ||||
|     // DEBUG / TEST: | ||||
|     if (QFileInfo(getUSBMountPath()).isDir()) | ||||
|         return true; | ||||
|     else | ||||
|         return false; | ||||
| #endif | ||||
|  | ||||
|     if (getUSBMountPath().isEmpty()) { | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     return true; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| /** | ||||
|  * @brief SupportSystem::getUSBMountPath | ||||
|  * @return path where a USB storage device is mounted. | ||||
|  * | ||||
|  * Note, do not return an empty string ("") here because a calling method (which could not be | ||||
|  * identified till now) is relying on this! | ||||
|  */ | ||||
| QString System::getUSBMountPath() | ||||
| { | ||||
|  | ||||
| #if defined (ARCH_DesktopLinux) | ||||
|     // DEBUG / TEST: | ||||
|     return QDir::homePath().append("/APconfigTest/USB"); | ||||
| #endif | ||||
|  | ||||
|     QProcess process; | ||||
|     process.setProcessChannelMode(QProcess::MergedChannels); | ||||
|  | ||||
|     QStringList mountLine; | ||||
|  | ||||
|     qDebug() << "System::getUSBMountPath()"; | ||||
|  | ||||
|     QRegExp devRegExp   = QRegExp("dev/sd*",   Qt::CaseSensitive, QRegExp::WildcardUnix); | ||||
|     QRegExp mountRegExp = QRegExp("media/sd*", Qt::CaseSensitive, QRegExp::WildcardUnix); | ||||
|  | ||||
|     QString commandString = "mount"; | ||||
|  | ||||
|     process.start(commandString); | ||||
|     if (!process.waitForStarted()) { | ||||
|         errorMsg = "System::getUSBMountPath(): ERROR: waitForStarted()"; | ||||
|         return ""; | ||||
|     } | ||||
|     if (!process.waitForFinished(600000)) { | ||||
|         errorMsg = "System::getUSBMountPath(): ERROR: " + process.errorString(); | ||||
|         qDebug() << errorMsg; | ||||
|         return ""; | ||||
|     } | ||||
|     else { | ||||
|         QByteArray bytes = process.readAll(); | ||||
|         QStringList lines = QString(bytes).split("\n"); | ||||
|         foreach (QString line, lines) { | ||||
|             qDebug() << "System::getUSBMountPath() line:    " << line; | ||||
|  | ||||
|             if (line.contains(devRegExp) && line.contains(mountRegExp)) { | ||||
|  | ||||
|                 qDebug() << "  -> this line is a usb storage device mount" << line; | ||||
|  | ||||
|                 mountLine = line.split(' '); | ||||
|                 if (mountLine.size() > 3) { | ||||
|                     qDebug() << "System::getUSBMountPath(): " << mountLine.at(0) << " is mounted on " << mountLine.at(2); | ||||
|                     return mountLine.at(2); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     qDebug() << "System::getUSBMountPath() no mounted usb device found!"; | ||||
|  | ||||
|     return ""; | ||||
| } | ||||
|  | ||||
|  | ||||
| /******************************************************************************** | ||||
|  * static function to check if a mounted sd-card is writable. | ||||
|  * | ||||
|  * Note: the caller must ensure (e.g. by calling checkForSDCard()) that a | ||||
|  * sd-card is mounted. | ||||
|  * | ||||
|  */ | ||||
| bool System::isSDCardWritable() | ||||
| { | ||||
|     QFileInfo fi(getSDCardMountPath()); | ||||
|  | ||||
|     if(fi.isDir() && fi.isWritable()) { | ||||
|         return true; | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         return false; | ||||
|     } | ||||
| } | ||||
|  | ||||
| /******************************************************************************** | ||||
|  * static function to check if a mounted usb-stick is writable. | ||||
|  * | ||||
|  * Note: the caller must ensure (e.g. by calling checkForUSBStick()) that a | ||||
|  * usb-stick is mounted. | ||||
|  * | ||||
|  */ | ||||
| bool System::isUSBStickWritable() | ||||
| { | ||||
|     QFileInfo fi(getUSBMountPath()); | ||||
|  | ||||
|     if(fi.isDir() && fi.isWritable()) { | ||||
|         return true; | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         return false; | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| /******************************************************************************** | ||||
|  * wrapper function for getting free space on a sd-card on ptu | ||||
|  * | ||||
|  */ | ||||
| quint32 System::getFreeDiskSpaceSDCard() | ||||
| { | ||||
| #if defined (ARCH_DesktopLinux) | ||||
|     return getFreeDiskSpace("/home/siegert/server1home"); | ||||
| #endif | ||||
|     return getFreeDiskSpace(getSDCardMountPath()); | ||||
| } | ||||
|  | ||||
|  | ||||
| /******************************************************************************** | ||||
|  * wrapper function for getting free space on a usb storage device on ptu | ||||
|  * | ||||
|  */ | ||||
| quint32 System::getFreeDiskSpaceUSBStick() | ||||
| { | ||||
| #if defined (ARCH_DesktopLinux) | ||||
|     return getFreeDiskSpace("/home/siegert/server1home"); | ||||
| #endif | ||||
|     return getFreeDiskSpace(getUSBMountPath()); | ||||
| } | ||||
|  | ||||
|  | ||||
| /******************************************************************************** | ||||
|  * static function to read free disk space | ||||
|  * | ||||
|  * example output of 'df' command on ptu: | ||||
|  *   # df | ||||
|  *   Filesystem     1K-blocks  Used Available Use% Mounted on | ||||
|  *   ubi0:rootfs       111704 78940     32764  71% / | ||||
|  *   devtmpfs           62072     0     62072   0% /dev | ||||
|  *   tmpfs              62332     0     62332   0% /dev/shm | ||||
|  *   tmpfs              62332   416     61916   1% /tmp | ||||
|  *   tmpfs              62332    24     62308   1% /run | ||||
|  *   ubi0:data         222260  5892    216368   3% /opt | ||||
|  *   tmpfs              62332     0     62332   0% /media | ||||
|  *   /dev/mmcblk0p1   3878912 87360   3791552   3% /media/mmcblk0p1 | ||||
|  * | ||||
|  * in this case, if we want to get the free space on the sd-card the methode should | ||||
|  * be called: | ||||
|  *   getFreeDiskSpace("/dev/mmcblk0p1") | ||||
|  * the result should be: | ||||
|  *   3791552 | ||||
|  * | ||||
|  */ | ||||
| quint32  System::getFreeDiskSpace(const QString& dev) | ||||
| { | ||||
|     quint32 availableSpace = 0; | ||||
|     QProcess process; | ||||
|     process.setProcessChannelMode(QProcess::MergedChannels); | ||||
|  | ||||
|     //qDebug() << "System::getFreeDiskSpace() dev = " << dev; | ||||
|  | ||||
|     QString commandString = "df"; | ||||
|  | ||||
|     process.start(commandString); | ||||
|     if (!process.waitForStarted()) { | ||||
|         errorMsg = "System::getFreeDiskSpace(): ERROR: waitForStarted()"; | ||||
|         return availableSpace; | ||||
|     } | ||||
|     if (!process.waitForFinished(600000)) { | ||||
|         errorMsg = "System::getFreeDiskSpace(): ERROR: " + process.errorString(); | ||||
|         qDebug() << errorMsg; | ||||
|         return availableSpace; | ||||
|     } | ||||
|     else { | ||||
|         QByteArray bytes = process.readAll(); | ||||
|         QStringList lines = QString(bytes).split("\n"); | ||||
|         foreach (QString line, lines) { | ||||
|             //qDebug() << "System::getFreeDiskSpace() line:    " << line; | ||||
|  | ||||
|  | ||||
|  | ||||
|             if (line.contains(dev)) { | ||||
|                 //qDebug() << "System::getFreeDiskSpace() line contains dev"; | ||||
|  | ||||
|                 QStringList processResultLine = line.split(' ', QString::SkipEmptyParts); | ||||
|                 if (processResultLine.size() >= 4) { | ||||
|                     bool ok; | ||||
|                     availableSpace = processResultLine.at(3).toLong(&ok); | ||||
|                     if (!ok) { | ||||
|                         return 0; | ||||
|                     } | ||||
|                     qDebug() << "System::getFreeDiskSpace() availableSpace = " << availableSpace; | ||||
|                 } | ||||
|                 else { | ||||
|                     return availableSpace; | ||||
|                 } | ||||
|  | ||||
|  | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     return availableSpace; | ||||
|  | ||||
|  | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| /******************************************************************************** | ||||
|  * static function to create a backup of logfiles | ||||
|  * | ||||
|  * 'targzfile' is an absolute filename for ar tar.gz file wich will be created | ||||
|  * 'filelistfile' is an absolute filename of a file containing all names of files | ||||
|  *                which should be backed up. | ||||
|  * | ||||
|  * returns: | ||||
|  *  1 on success | ||||
|  *  0 on failure | ||||
|  * | ||||
|  */ | ||||
| quint8 System::createLogFileBackup(const QString & targzfile, const QString &filelistfile) | ||||
| { | ||||
|     QProcess process; | ||||
|     process.setProcessChannelMode(QProcess::MergedChannels); | ||||
|  | ||||
|     qDebug() << "System::createLogFileBackup() targzfile    = " << targzfile; | ||||
|     qDebug() << "System::createLogFileBackup() filelistfile = " << filelistfile; | ||||
|  | ||||
|     QString commandString = "tar hczvf " + targzfile + " -T " + filelistfile; | ||||
|  | ||||
|     qDebug() << "System::createLogFileBackup() tar command:" << commandString; | ||||
|  | ||||
|     process.start(commandString); | ||||
|     if (!process.waitForStarted()) { | ||||
|         errorMsg = "System::createLogFileBackup(): ERROR: waitForStarted()"; | ||||
|         return 1; | ||||
|     } | ||||
|     if (!process.waitForFinished(600000)) { | ||||
|         errorMsg = "System::createLogFileBackup(): ERROR: " + process.errorString(); | ||||
|         qDebug() << errorMsg; | ||||
|         return 1; | ||||
|     } | ||||
|     else { | ||||
|         // DEBUG | ||||
|         //QByteArray bytes = process.readAll(); | ||||
|         //QStringList lines = QString(bytes).split("\n"); | ||||
|         //foreach (QString line, lines) { | ||||
|         //    qDebug() << "System::createLogFileBackup() line:    " << line; | ||||
|         //} | ||||
|     } | ||||
|  | ||||
|     QString resultMessage; | ||||
|     QProcess::ExitStatus exitStatus = process.exitStatus(); | ||||
|     int exitCode = process.exitCode(); | ||||
|  | ||||
|     if (exitStatus == QProcess::CrashExit) { | ||||
|         resultMessage = "local backup program crashed"; | ||||
|         qCritical() << resultMessage; | ||||
|         return 1; | ||||
|     } | ||||
|     else | ||||
|  | ||||
|         /** | ||||
|          *  note for exit codes of process tar: | ||||
|          *  see: http://www.gnu.org/software/tar/manual/html_section/tar_19.html#Synopsis | ||||
|          * | ||||
|          *  tar returns exitCode 2 if some files in logFileList could not be read (e.g. if they are not existing) | ||||
|          *  however, tar-archive is created although! | ||||
|          *  This means success for our needs! | ||||
|          */ | ||||
|  | ||||
|     if ( (exitCode != 0) && (exitCode != 2) && (exitStatus == QProcess::NormalExit) ) { | ||||
|         resultMessage = "Local Backup file \"" + targzfile + "\" with exit code = " + QString::number(exitCode); | ||||
|         qCritical() << resultMessage; | ||||
|         return 1; | ||||
|     } | ||||
|     else { | ||||
|         resultMessage = "Local Backup file \"" + targzfile + "\" was created successfully."; | ||||
|         qCritical() << resultMessage; | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     return 0; | ||||
|  | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| /***************************************************************************** | ||||
|  * get SD-card mount path. | ||||
|  */ | ||||
| QString System::getSDCardMountPath() | ||||
| { | ||||
| #if defined (ARCH_PTU4) | ||||
|     return "/media/mmcblk0p1"; | ||||
| #elif defined (ARCH_PTU5) | ||||
|     return "/media/mmcblk0p1"; | ||||
| #elif defined (ARCH_DesktopLinux) | ||||
|     // DEBUG / TEST: | ||||
|     return "/home/siegert/APconfigTest"; | ||||
| #endif | ||||
|     return ""; | ||||
| } | ||||
|  | ||||
|  | ||||
| /***************************************************************************** | ||||
|  * get SD-card device name. | ||||
|  */ | ||||
| QString System::getSDCardDeviceName() | ||||
| { | ||||
| #if defined (ARCH_PTU4) | ||||
|     return "/dev/mmcblk0p1"; | ||||
| #elif defined (ARCH_PTU5) | ||||
|     return "/dev/mmcblk0p1"; | ||||
| #elif defined (ARCH_DesktopLinux) | ||||
|     return "/dev/sdf1"; | ||||
| #endif | ||||
|     return ""; | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| /******************************************************************************** | ||||
|  * static function to build and return a unique device id. | ||||
|  * This id consists currently only of the mac-address of network device eth0. | ||||
|  */ | ||||
| QString System::getUniqueDeviceId() | ||||
| { | ||||
|     QString filename = "/sys/class/net/eth0/address"; | ||||
|     QString result; | ||||
|  | ||||
|     QFile file(filename); | ||||
|     if(!file.open(QIODevice::ReadOnly)) { | ||||
|         qCritical() << "System::getUniqueDeviceId() could not open file: " << filename; | ||||
|         result = "0"; | ||||
|     } | ||||
|     else { | ||||
|         QTextStream in(&file); | ||||
|         result = in.readLine(20); | ||||
|     } | ||||
|  | ||||
|     file.close(); | ||||
|  | ||||
|     result.remove(':'); | ||||
|  | ||||
|     return result; | ||||
|  | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| QString System::readStringFromFile(const QString & filename) | ||||
| { | ||||
| #if defined (ARCH_PTU4) | ||||
|     QFileInfo fileinfo(filename); | ||||
| #else | ||||
|     QFileInfo fileinfo(QDir::homePath() + filename); | ||||
| #endif | ||||
|     if (! fileinfo.isReadable() ) { | ||||
|         qDebug() << "System::readStringFromFile(): \"" << filename << "\" is not readable"; | ||||
|         return ""; | ||||
|     } | ||||
|  | ||||
| #if defined (ARCH_PTU4) | ||||
|     QFile file(filename); | ||||
| #else | ||||
|     QFile file(QDir::homePath() + filename); | ||||
| #endif | ||||
|     if (!file.open(QIODevice::ReadOnly | QIODevice::Text)) { | ||||
|         qDebug() << "System::readStringFromFile() cannot open file: " << filename; | ||||
|         return ""; | ||||
|     } | ||||
|  | ||||
|     QTextStream in(&file); | ||||
|  | ||||
|     QString stringValue = in.readLine(100); | ||||
|     qDebug() << "System::readStringFromFile() stringValue = " << stringValue; | ||||
|  | ||||
|     file.close(); | ||||
|  | ||||
|     return stringValue; | ||||
| } | ||||
|  | ||||
|  | ||||
| int System::readIntFromFile(const QString & filename) | ||||
| { | ||||
|     bool ok; | ||||
|     int result = (int)readStringFromFile(filename).toInt(&ok); | ||||
|  | ||||
|     if (!ok) { | ||||
|         result = 0; | ||||
|     } | ||||
|  | ||||
|     return result; | ||||
| } | ||||
|  | ||||
|  | ||||
| /**************************************************************************** | ||||
|  * read ptu internal data | ||||
|  * | ||||
|  */ | ||||
| QString System::getPTU4SerialNumber() | ||||
| { | ||||
|     return readStringFromFile("/sys/bus/i2c/devices/0-0050/SNO"); | ||||
| } | ||||
|  | ||||
| QString System::getPTU4MACAddress() | ||||
| { | ||||
|     return readStringFromFile("/sys/bus/i2c/devices/0-0050/MAC"); | ||||
| } | ||||
|  | ||||
							
								
								
									
										97
									
								
								common/src/message_handler.cpp
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										97
									
								
								common/src/message_handler.cpp
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,97 @@ | ||||
| #include "message_handler.h" | ||||
|  | ||||
| #include <QDateTime> | ||||
| #include <cstring> | ||||
| #include <QString> | ||||
| #include <QFileInfo> | ||||
| #include <QMessageLogContext> | ||||
|  | ||||
|  | ||||
| static char const *DBG_NAME[] = { "DBG  ", "WARN ", "CRIT ", "FATAL", "INFO " }; | ||||
| static bool installedMsgHandler = false; | ||||
| static int debugLevel = LOG_NOTICE; | ||||
|  | ||||
| int getDebugLevel() { return debugLevel; } | ||||
| void setDebugLevel(int newDebugLevel) { | ||||
|     debugLevel = newDebugLevel; | ||||
| } | ||||
|  | ||||
| bool messageHandlerInstalled() { | ||||
|     return installedMsgHandler; | ||||
| } | ||||
|  | ||||
| QtMessageHandler atbInstallMessageHandler(QtMessageHandler handler) { | ||||
|     installedMsgHandler = (handler != 0); | ||||
|     static QtMessageHandler prevHandler = nullptr; | ||||
|     if (handler) { | ||||
|         prevHandler = qInstallMessageHandler(handler); | ||||
|         return prevHandler; | ||||
|     } else { | ||||
|         return qInstallMessageHandler(prevHandler); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// | ||||
| /// \brief Print message according to given debug level. | ||||
| /// | ||||
| /// \note Install this function using qInstallMsgHandler(). | ||||
| /// | ||||
| ///  int main(int argc, char **argv) { | ||||
| ///     installMsgHandler(atbDebugOutput); | ||||
| ///     QApplication app(argc, argv); | ||||
| ///     ... | ||||
| ///     return app.exec(); | ||||
| ///  } | ||||
| /// | ||||
| #if (QT_VERSION > QT_VERSION_CHECK(5, 0, 0) && QT_VERSION < QT_VERSION_CHECK(6, 0, 0)) | ||||
| void atbDebugOutput(QtMsgType type, const QMessageLogContext &context, const QString &msg) { | ||||
|     Q_UNUSED(context); | ||||
|     QString const localMsg = QString(DBG_NAME[type]) + msg.toLocal8Bit(); | ||||
|  | ||||
|     switch (debugLevel) { | ||||
|         case LOG_DEBUG: { // debug-level message | ||||
|             syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|         } break; | ||||
|         case LOG_INFO: { // informational message | ||||
|             if (type != QtDebugMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         case LOG_NOTICE: { // normal, but significant, condition | ||||
|             if (type != QtDebugMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         case LOG_WARNING: { // warning conditions | ||||
|             if (type != QtInfoMsg && type != QtDebugMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         case LOG_ERR: { // error conditions | ||||
|             if (type != QtInfoMsg && type != QtDebugMsg && type != QtWarningMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         case LOG_CRIT: { // critical conditions | ||||
|             if (type != QtInfoMsg && type != QtDebugMsg && type != QtWarningMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         case LOG_ALERT: { // action must be taken immediately | ||||
|             if (type != QtInfoMsg && type != QtDebugMsg && type != QtWarningMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         case LOG_EMERG: { // system is unusable | ||||
|             if (type != QtInfoMsg && type != QtDebugMsg && type != QtWarningMsg) { | ||||
|                 syslog(LOG_DEBUG, "%s", localMsg.toStdString().c_str()); | ||||
|             } | ||||
|         } break; | ||||
|         default: { | ||||
|             //fprintf(stderr, "%s No ErrorLevel defined! %s\n", | ||||
|             //    datetime.toStdString().c_str(), msg.toStdString().c_str()); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| #endif | ||||
|  | ||||
							
								
								
									
										2372
									
								
								interfaces.h
									
									
									
									
									
								
							
							
						
						
									
										2372
									
								
								interfaces.h
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										871
									
								
								update.cpp
									
									
									
									
									
								
							
							
						
						
									
										871
									
								
								update.cpp
									
									
									
									
									
								
							| @@ -1,871 +0,0 @@ | ||||
| #include "update.h" | ||||
| #include "worker.h" | ||||
| #include "utils.h" | ||||
| #include "update_dc_event.h" | ||||
| #include "mainwindow.h" | ||||
|  | ||||
| #include <QCoreApplication> | ||||
| #include <QApplication> | ||||
| #include <QFile> | ||||
| #include <QTemporaryFile> | ||||
| #include <QDebug> | ||||
| #include <QTextStream> | ||||
| #include <QRegularExpression> | ||||
| #include <QRegExp> | ||||
| #include <QApplication> | ||||
|  | ||||
| #if defined (Q_OS_UNIX) || defined (Q_OS_LINUX) | ||||
| #include "unistd.h" | ||||
| #endif | ||||
|  | ||||
| #include "plugins/interfaces.h" | ||||
|  | ||||
| #include <QSharedMemory> | ||||
| #include <QScopedPointer> | ||||
| #include <QDir> | ||||
| #include <QThread> | ||||
| #include <QDateTime> | ||||
| #include <QPluginLoader> | ||||
| #include <QMap> | ||||
|  | ||||
| #define UPDATE_OPKG                 (1) | ||||
| #define UPDATE_DC                   (0) | ||||
|  | ||||
| static const QMap<QString, int> baudrateMap = { | ||||
|   {"1200"   ,   0}, {"9600"   ,   1}, {"19200"  ,   2}, {"38400"  ,   3}, | ||||
|   {"57600"  ,   4}, {"115200" ,   5} | ||||
| }; | ||||
|  | ||||
| QPluginLoader Update::pluginLoader; | ||||
|  | ||||
| hwinf *Update::loadDCPlugin(QDir const &plugInDir, QString const &fname) { | ||||
|     hwinf *hw = nullptr; | ||||
|     if (plugInDir.exists()) { | ||||
|         QString pluginLibName(fname); | ||||
|         pluginLibName = plugInDir.absoluteFilePath(pluginLibName); | ||||
|         QFileInfo info(pluginLibName); | ||||
|         if (info.exists()) { | ||||
|             pluginLibName = plugInDir.absoluteFilePath(pluginLibName); | ||||
|             pluginLoader.setFileName(pluginLibName); | ||||
|             // static QPluginLoader pluginLoader(pluginLibName); | ||||
|             if (!pluginLoader.load()) { | ||||
|                 qCritical() << "in directory" << plugInDir.absolutePath(); | ||||
|                 qCritical() << "cannot load plugin" << pluginLoader.fileName(); | ||||
|                 qCritical() << pluginLoader.errorString(); | ||||
|                 exit(-1); | ||||
|             } | ||||
|             if (!pluginLoader.isLoaded()) { | ||||
|                 qCritical() << pluginLoader.errorString(); | ||||
|                 exit(-2); | ||||
|             } | ||||
|             QObject *plugin = pluginLoader.instance(); | ||||
|             if (!plugin) { | ||||
|                 qCritical() << "cannot start instance"; | ||||
|                 exit(-3); | ||||
|             } | ||||
|             if (! (hw = qobject_cast<hwinf *>(plugin))) { | ||||
|                 qCritical() << "cannot cast plugin" << plugin << "to hwinf"; | ||||
|                 exit(-4); | ||||
|             } | ||||
|         } else { | ||||
|             qCritical() << pluginLibName << "does not exist"; | ||||
|             exit(-5); | ||||
|         } | ||||
|     } else { | ||||
|         qCritical() << "plugins directory" << plugInDir.absolutePath() | ||||
|                     << "does not exist"; | ||||
|         exit(-6); | ||||
|     } | ||||
|     return hw; | ||||
| } | ||||
|  | ||||
| bool Update::unloadDCPlugin() { | ||||
|     if (pluginLoader.unload()) { | ||||
|         qCritical() << "unloaded plugin" << pluginLoader.fileName(); | ||||
|         // Note: will re-instantiate the library ! | ||||
|         // QObject *rootObject = pluginLoader.instance(); | ||||
|         // if (rootObject) { | ||||
|         //    qCritical() << "reloaded plugin: root object again available"; | ||||
|         //    return false; | ||||
|         // } | ||||
|         // qCritical()unloaded plugin: root object gone"; | ||||
|         return true; | ||||
|     } | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| Update::Update(Worker *worker, | ||||
|                QString customerRepository, | ||||
|                QString customerNrStr, | ||||
|                QString branchName, | ||||
|                QString plugInDir, | ||||
|                QString pluginName, | ||||
|                QString workingDir, | ||||
|                bool dryRun, | ||||
|                QObject *parent, | ||||
|                char const *serialInterface, | ||||
|                char const *baudrate) | ||||
|     : QObject(parent) | ||||
|     , m_hw(loadDCPlugin(QDir(plugInDir), pluginName)) | ||||
|     , m_worker(worker) | ||||
|     , m_serialInterface(serialInterface) | ||||
|     , m_baudrate(baudrate) | ||||
|     , m_customerRepository(customerRepository) | ||||
|     , m_customerNrStr(customerNrStr) | ||||
|     , m_branchName(branchName) | ||||
|     , m_pluginName(pluginName) | ||||
|     , m_workingDir(workingDir) | ||||
|     , m_dryRun(dryRun) { | ||||
|  | ||||
|     qInfo() << "UPDATE: m_serialInterface    ..." << m_serialInterface; | ||||
|     qInfo() << "UPDATE: m_baudrate           ..." << m_baudrate; | ||||
|     qInfo() << "UPDATE: m_customerRepository ..." << m_customerRepository; | ||||
|     qInfo() << "UPDATE: m_customerNr ..........." << m_customerNrStr; | ||||
|     qInfo() << "UPDATE: m_branchName ..........." << m_branchName; | ||||
|     qInfo() << "UPDATE: m_pluginName ..........." << m_pluginName; | ||||
|     qInfo() << "UPDATE: m_workingDirectory ....." << m_workingDir; | ||||
| } | ||||
|  | ||||
| Update::~Update() { | ||||
| } | ||||
|  | ||||
| Update::DownloadResult Update::sendStatus(int ret) const { | ||||
|     switch (ret) {                      // return values of dc are: | ||||
|     case 0:                             // 0: no answer by now | ||||
|         return DownloadResult::NOP;     // 1: error | ||||
|     case 10:                            // 10: success | ||||
|         return DownloadResult::OK; | ||||
|     default:; | ||||
|     } | ||||
|     return DownloadResult::ERROR; | ||||
| } | ||||
|  | ||||
| Update::DownloadResult Update::sendNextAddress(int bNum) const { | ||||
|     // sends address only if blockNumber is one of 0, 1024, 2048, 3072, 4096 | ||||
|     int noAnswerCount = 0; | ||||
|     int errorCount = 0; | ||||
|     if ( bNum==0 || bNum==1024 || bNum==2048 || bNum==3072 || bNum==4096 ) { | ||||
|         // qDebug() << "addr-block" << bNum << "..."; | ||||
|         while (noAnswerCount <= 250) { | ||||
|             m_hw->bl_sendAddress(bNum); | ||||
|             QThread::msleep(100); | ||||
|             DownloadResult const res = sendStatus(m_hw->bl_wasSendingAddOK()); | ||||
|             if (res != DownloadResult::NOP) { | ||||
|                 if (res == DownloadResult::ERROR) { | ||||
|                     if (++errorCount >= 10) { | ||||
|                         qCritical() << "addr-block" << bNum << "...FAILED"; | ||||
|                         return res; | ||||
|                     } | ||||
|                 } else { // res == DownloadResult::OK | ||||
|                     // qInfo() << "addr-block" << bNum << "...OK"; | ||||
|                     return res; | ||||
|                 } | ||||
|             } else { | ||||
|                 noAnswerCount += 1; // no answer by now | ||||
|             } | ||||
|         } | ||||
|         // wait max. about 3 seconds | ||||
|         return DownloadResult::TIMEOUT; | ||||
|     } | ||||
|     // blockNumber is not one of 0, 1024, 2048, 3072, 4096 -> do nothing | ||||
|     return DownloadResult::NOP; | ||||
| } | ||||
|  | ||||
| Update::DownloadResult Update::sendNextDataBlock(QByteArray const &binary, | ||||
|                                                  int bNum) const { | ||||
|     uint8_t local[66]; | ||||
|     int const bAddr = bNum * 64; | ||||
|     int noAnswerCount = 0; | ||||
|     int errorCount = 0; | ||||
|  | ||||
|     memcpy(local, binary.constData() + bAddr, 64); | ||||
|     local[64] = local[65] = 0x00; | ||||
|  | ||||
|     // QByteArray b((const char *)(&local[0]), 64); | ||||
|     // qCritical() << "SNDB" << bNum << b.size() << b.toHex(); | ||||
|  | ||||
|     while (noAnswerCount <= 250) { | ||||
|         m_hw->bl_sendDataBlock(64, local); | ||||
|         QThread::msleep(10); | ||||
|         DownloadResult const res = sendStatus(m_hw->bl_wasSendingDataOK()); | ||||
|         if (res != DownloadResult::NOP) { | ||||
|             if (res == DownloadResult::ERROR) { | ||||
|                 if (++errorCount >= 10) { | ||||
|                     qCritical() << "data for block" << bNum << "...FAILED"; | ||||
|                     return res; | ||||
|                 } | ||||
|             } else { | ||||
|                 // qInfo() << "data for block" << bNum << "OK"; | ||||
|                 return res; | ||||
|             } | ||||
|         } else { | ||||
|             noAnswerCount += 1; // no answer by now | ||||
|         } | ||||
|     } | ||||
|     // wait max. about 3 seconds | ||||
|     return DownloadResult::TIMEOUT; | ||||
| } | ||||
|  | ||||
| Update::DownloadResult Update::dc_downloadBinary(QByteArray const &b) const { | ||||
|     int const nBlocks = (((b.size())%64)==0) ? (b.size()/64) : (b.size()/64)+1; | ||||
|  | ||||
|     // fill lst block of data to be sent with 0xFF | ||||
|     QByteArray ba = b.leftJustified(nBlocks*64, (char)(0xFF)); | ||||
|  | ||||
|     qInfo() << "total number of bytes to send to dc" << ba.size(); | ||||
|     qInfo() << "total number of blocks to send to dc" << nBlocks; | ||||
|  | ||||
|     int bNum = 0; | ||||
|     DownloadResult res = DownloadResult::OK; | ||||
|     fprintf(stderr, "\n64-byte block %04d ", bNum); | ||||
|     while (res != DownloadResult::ERROR &&  bNum < nBlocks) { | ||||
|         if ((res = sendNextAddress(bNum)) != DownloadResult::ERROR) { | ||||
|             if ((res = sendNextDataBlock(ba, bNum)) != DownloadResult::ERROR) { | ||||
|                 bNum += 1; | ||||
|                 fprintf(stderr, "."); | ||||
|                 if ((bNum % 80) == 0) { | ||||
|                     fprintf(stderr, "\n64-byte block %04d ", bNum); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     fprintf(stderr, "\nlast 64-byte block %04d\n", bNum); | ||||
|  | ||||
|     int const rest = ba.size() % 64; | ||||
|     int const offset = ba.size() - rest; | ||||
|     char const *startAddress = ba.constData() + offset; | ||||
|  | ||||
|     if (rest > 0) { | ||||
|         // SHOULD NEVER HAPPEN !!! | ||||
|         uint8_t local[66]; | ||||
|         memset(local, 0xFF, sizeof(local)); | ||||
|         memcpy(local, startAddress, rest); | ||||
|         qCritical() << "ERROR SEND REMAINING" << rest << "BYTES"; | ||||
|         m_hw->bl_sendDataBlock(64, local); | ||||
|     } | ||||
|  | ||||
|     m_hw->bl_sendLastBlock(); | ||||
|     qInfo() << "last result" << (int)sendStatus(m_hw->bl_wasSendingDataOK()); | ||||
|     return res; | ||||
| } | ||||
|  | ||||
| bool Update::startBootloader() const { // deprecated | ||||
|     return false; | ||||
| #if 0 | ||||
|     int nStartTry = 5; | ||||
|     while (--nStartTry >= 0) { | ||||
|         m_hw->bl_startBL(); | ||||
|         QThread::msleep(500); | ||||
|         int nCheckTry = 10; | ||||
|         while (--nCheckTry >= 0) { | ||||
|             m_hw->bl_checkBL(); | ||||
|             QThread::msleep(500); | ||||
|             if (m_hw->bl_isUp()) { | ||||
|                 qInfo() << "starting bootloader...OK"; | ||||
|                 return true; | ||||
|             } else { | ||||
|                 qCritical() << "bootloader not up (" | ||||
|                             << nStartTry << "," << nCheckTry << ")" << QThread::currentThread(); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     qCritical() << "starting bootloader...FAILED" << QThread::currentThread(); | ||||
|     return false; | ||||
| #endif | ||||
| } | ||||
|  | ||||
| bool Update::stopBootloader() const { | ||||
|     // stop bootloader: this MUST work -> otherwise the PSA has to be restarted | ||||
|     // manually | ||||
|     emit m_worker->showErrorMessage("dc update", "stopping bootloader..."); | ||||
|  | ||||
|     int nTryFinalize = 1; // could do this in an endless loop | ||||
|  | ||||
|     do { | ||||
|         // in principle, any value except BL_STOP will do, as we want to detect | ||||
|         // change to BL_STOP | ||||
|         m_worker->mainWindow()->setUpdateStep(UpdateDcEvent::UpdateStep::BL_CHECK); | ||||
|  | ||||
|         QApplication::postEvent( | ||||
|             m_worker->mainWindow(), | ||||
|             new UpdateDcEvent(m_worker, UpdateDcEvent::UpdateStep::BL_STOP, nTryFinalize)); | ||||
|  | ||||
|         QThread::sleep(1); | ||||
|  | ||||
|         int const cntLimit = 20; | ||||
|         int cnt = 0; | ||||
|         while (++cnt < cntLimit && | ||||
|             m_worker->mainWindow()->updateStep() != UpdateDcEvent::UpdateStep::BL_STOP) { | ||||
|             // wait until bl_stopBL() has been sent | ||||
|             QThread::msleep(500); | ||||
|         } | ||||
|  | ||||
|         QApplication::postEvent( | ||||
|             m_worker->mainWindow(), | ||||
|             new UpdateDcEvent(m_worker, UpdateDcEvent::UpdateStep::BL_CHECK, nTryFinalize)); | ||||
|         QThread::sleep(1); | ||||
|  | ||||
|         QApplication::postEvent( | ||||
|             m_worker->mainWindow(), | ||||
|             new UpdateDcEvent(m_worker, UpdateDcEvent::UpdateStep::BL_IS_UP, nTryFinalize)); | ||||
|         QThread::sleep(1); | ||||
|  | ||||
|         cnt = 0; | ||||
|         while (++cnt < cntLimit && | ||||
|             m_worker->mainWindow()->updateStep() != UpdateDcEvent::UpdateStep::BL_IS_DOWN) { | ||||
|             // wait until  done | ||||
|             QThread::msleep(200); | ||||
|         } | ||||
|  | ||||
|     } while (++nTryFinalize <= MainWindow::BL_STOP_COUNT && | ||||
|              m_worker->mainWindow()->updateStep() != UpdateDcEvent::UpdateStep::BL_IS_DOWN); | ||||
|  | ||||
|     return (m_worker->mainWindow()->updateStep() == UpdateDcEvent::UpdateStep::BL_IS_DOWN); | ||||
| } | ||||
|  | ||||
| // br is a index into a table, used for historical reasons. | ||||
| bool Update::openSerial(int br, QString baudrate, QString comPort) const { | ||||
|     qDebug() << "opening serial" << br << baudrate << comPort << "..."; | ||||
|     if (m_hw->dc_openSerial(br, baudrate, comPort, 1) == true) { // 1 for connect | ||||
|         Utils::printInfoMsg( | ||||
|             QString("OPENING SERIAL %1").arg(br) | ||||
|                   + " " + baudrate + " " + comPort + "...OK"); | ||||
|  | ||||
|         // m_hw->dc_autoRequest(true); | ||||
|         m_hw->dc_autoRequest(false); | ||||
|         QThread::sleep(1); | ||||
|  | ||||
|         Utils::printInfoMsg(QString("IS PORT OPEN %1").arg(m_hw->dc_isPortOpen())); | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|     Utils::printCriticalErrorMsg( | ||||
|         QString("OPENING SERIAL %1").arg(br) | ||||
|               + " " + baudrate + " " + comPort + "...FAILED"); | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| void Update::closeSerial() const { | ||||
|     qInfo() << "CLOSED SERIAL" << m_baudrate << m_serialInterface; | ||||
|     m_hw->dc_closeSerial(); | ||||
| } | ||||
|  | ||||
| bool Update::isSerialOpen() const { | ||||
|     return m_hw->dc_isPortOpen(); | ||||
| } | ||||
|  | ||||
| bool Update::resetDeviceController() const { // deprecated | ||||
|     return false; | ||||
| #if 0 | ||||
|     qDebug() << "resetting device controller..."; | ||||
|     m_hw->bl_rebootDC(); | ||||
|     // wait maximally 3 seconds, before starting bootloader | ||||
|     qInfo() << "resetting device controller...OK"; | ||||
|     return true; | ||||
| #endif | ||||
| } | ||||
|  | ||||
| QByteArray Update::loadBinaryDCFile(QString filename) const { | ||||
|     qDebug() << "loading dc binary" << filename << "..."; | ||||
|  | ||||
|     QFile file(filename); // closed in destructor call | ||||
|     if (!file.exists()) { | ||||
|         qCritical() << file.fileName() << "does not exist"; | ||||
|         return QByteArray(); | ||||
|     } | ||||
|     if (!file.open(QIODevice::ReadOnly)) { | ||||
|         qCritical() << "cannot open file" << file.fileName(); | ||||
|         return QByteArray(); | ||||
|     } | ||||
|     qInfo() << "loading dc binary" << filename << "...OK"; | ||||
|     return file.readAll(); | ||||
| } | ||||
|  | ||||
| bool Update::downloadBinaryToDC(QString const &bFile) const { | ||||
|     qDebug() << "sending" << bFile << "to dc..."; | ||||
|     QByteArray const dcBinary = loadBinaryDCFile(bFile); | ||||
|     if (dcBinary.size() > 0) { | ||||
|         if (dc_downloadBinary(dcBinary) != DownloadResult::OK) { | ||||
|             qCritical() << "sending" << bFile << "to dc...FAILED"; | ||||
|             return false; | ||||
|         } else { | ||||
|             qInfo() << "sending" << bFile << "to dc...OK"; | ||||
|         } | ||||
|     } else { | ||||
|         qCritical() << "sending" << bFile << "to dc...FAILED"; | ||||
|         qCritical() << "loading binary" << bFile << "FAILED"; | ||||
|         return false; | ||||
|     } | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| /* | ||||
|  | ||||
|  /////////////////////////////////////////////////////////////////////////////// | ||||
|  // | ||||
|  //                        USING THE DC BOOTLOADER | ||||
|  // | ||||
|  /////////////////////////////////////////////////////////////////////////////// | ||||
|  | ||||
|  1  :   bl_reboot()     // send to application, want DC2 to reset (in order to | ||||
|                         // start the bootloader) | ||||
|                         // | ||||
|                         // NOTE: this function is NOT reliable !!! Sometimes it | ||||
|                         // simply does not work, in which case bl_startBL, | ||||
|                         // bl_checkBL and bl_isUp do not work as well. | ||||
|                         // Alas, there is no feedback if bl_reboot worked! | ||||
|                         // | ||||
|                         // NOTE: this function can be called only once per | ||||
|                         // minute, because once called again, the controller | ||||
|                         // performs some self-checks consuming some time. | ||||
|                         // | ||||
|                         // NOTE: after a successful bl_reboot(), the device is | ||||
|                         // waiting about 4 seconds in the bootloader. To stay in | ||||
|                         // the bootloader, we have to send the command | ||||
|                         // bl_startBL(), which is kind of a misnomer, as it | ||||
|                         // should be bl_doNotLeaveBL(). | ||||
|                         // | ||||
|  2  :   bl_startBL():   // send within 4s after DC power-on, otherwise | ||||
|                         // bootloader is left. | ||||
|                         // | ||||
|                         // NOTE: a running bootloader is a MUST for the download | ||||
|                         // process of a device controller firmware as it does | ||||
|                         // the actual writing of the memory (the bl_reboot() | ||||
|                         // from above erases the available memory). | ||||
|                         // | ||||
|  3  :   bl_check():     // send command to verify if bl is up | ||||
|                         // | ||||
|                         // NOTE: this command is kind of a request that we want | ||||
|                         // to check if the bootloader is up. The device | ||||
|                         // (actually the bootloader) responds with its version. | ||||
|                         // | ||||
|  4  :   bl_isUp():      // returns true if bl is up and running | ||||
|                         // | ||||
|                         // NOTE: we know what the bootloader version actually is | ||||
|                         // as the bootloader does not change. By comparing the | ||||
|                         // string received in the previous step with this known | ||||
|                         // version string we know if the bootloader is up. | ||||
|                         // | ||||
|                         // NOTE FOR ALL PREVIOUS STEPS: execute them in their | ||||
|                         // own slots each to be sure to receive any possible | ||||
|                         // responds from the device. | ||||
|                         // | ||||
|  5  :   bl_sendAddress(blockNumber) | ||||
|                         // send start address, nr of 64-byte block, start with 0 | ||||
|                         // will be sent only for following block-numbers: | ||||
|                         // 0, 1024, 2048, 3072 and 4096, so basically every | ||||
|                         // 64kByte. | ||||
|                         // for other addresses nothing happens | ||||
|  | ||||
|  6  :   bl_wasSendingAddOK() | ||||
|                         // return val:  0: no response by now | ||||
|                         //              1: error | ||||
|                         //             10: OK | ||||
|  | ||||
|  7  :   bl_sendDataBlock() | ||||
|                         // send 64 byte from bin file | ||||
|  | ||||
|  8  :   bl_sendLastBlock() | ||||
|                         // send this command after all data are transferred | ||||
|  | ||||
|  9  :   bl_wasSendingDataOK() | ||||
|                         // return val:  0: no response by now | ||||
|                         //              1: error | ||||
|                         //             10: OK | ||||
|  | ||||
|  10 :   bl_stopBL()     // leave bl and start (the new) application | ||||
|                         // | ||||
|                         // NOTE: this function MUST work under all conditions. | ||||
|                         // Alas, there is no direct result for this command, so | ||||
|                         // the only way of knowing it was successful is to ask | ||||
|                         // the device if the bootloader is still running. | ||||
|                         // There is no problem to repeat this command until the | ||||
|                         // bootloader is really not running anymore. | ||||
|  */ | ||||
| bool Update::updateBinary(char const *fileToSendToDC) { | ||||
|     qInfo() << "UPDATING DEVICE CONTROLLER FIRMWARE BINARY" << fileToSendToDC; | ||||
|     QFile fn(fileToSendToDC); | ||||
|     bool r; | ||||
|     if ((r = fn.exists()) == true) { | ||||
|         QFileInfo fi(fn); | ||||
|         if ((r = updateDC(fileToSendToDC)) == true) { | ||||
|             Utils::printInfoMsg( | ||||
|                 QString("      UPDATING BINARY ") + fi.fileName() | ||||
|                       + QString(" (size=%1").arg(fi.size()) + ") DONE"); | ||||
|         } else { | ||||
|             Utils::printCriticalErrorMsg( | ||||
|                 QString("      UPDATING BINARY ") + fi.fileName() | ||||
|                       + QString(" (size=%1").arg(fi.size()) + ") FAILED"); | ||||
|         } | ||||
|     } else { | ||||
|         Utils::printCriticalErrorMsg( | ||||
|             QString(fileToSendToDC) + " DOES NOT EXIST -> NO UPDATE OF DC FIRMWARE"); | ||||
|     } | ||||
|     return r; | ||||
| } | ||||
|  | ||||
| bool Update::updateDC(QString bFile) const { | ||||
|     qDebug() << "IN UPDATEDC: UPDATING DC: FILE TO SEND" << bFile; | ||||
|  | ||||
|     m_worker->mainWindow()->setUpdateStep(UpdateDcEvent::UpdateStep::NONE); | ||||
|  | ||||
|     QApplication::postEvent(            // step 1: reset device controller | ||||
|         m_worker->mainWindow(), | ||||
|         new UpdateDcEvent(m_worker, UpdateDcEvent::UpdateStep::DC_REBOOT, 1)); | ||||
|     QThread::sleep(1); | ||||
|  | ||||
|     for (int i=1; i <= MainWindow::BL_START_COUNT; ++i) { | ||||
|         QApplication::postEvent(        // step 2: start bootloader (5x) | ||||
|             m_worker->mainWindow(), | ||||
|             new UpdateDcEvent(m_worker, UpdateDcEvent::UpdateStep::BL_START, i)); | ||||
|         QThread::sleep(1); | ||||
|     } | ||||
|  | ||||
|     int const cntLimit = 100;       // wait until its for sure that bl_startBL() | ||||
|     int cnt = 0;                    // has been excuted | ||||
|     while (++cnt < cntLimit && | ||||
|         m_worker->mainWindow()->updateStep() != UpdateDcEvent::UpdateStep::BL_START) { | ||||
|         // wait until all bl_startBL() are done | ||||
|         QThread::msleep(200); | ||||
|     } | ||||
|  | ||||
|     if (cnt == cntLimit) { | ||||
|         // start events not received ??? | ||||
|         Utils::printCriticalErrorMsg("BL_START EVENT NOT RECEIVED AFTER 20 SECS"); | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     m_worker->mainWindow()->setUpdateStep(UpdateDcEvent::UpdateStep::BL_CHECK); | ||||
|  | ||||
|     for (int i=1; i <= MainWindow::BL_IS_UP_COUNT; ++i) { | ||||
|         QApplication::postEvent(m_worker->mainWindow(), new UpdateDcEvent(m_worker, UpdateDcEvent::UpdateStep::BL_CHECK, i)); | ||||
|         QThread::sleep(1); | ||||
|         QApplication::postEvent(m_worker->mainWindow(), new UpdateDcEvent(m_worker, UpdateDcEvent::UpdateStep::BL_IS_UP, i)); | ||||
|         if (m_worker->mainWindow()->updateStep() == UpdateDcEvent::UpdateStep::BL_IS_UP) { | ||||
|             break; | ||||
|         } | ||||
|         QThread::sleep(1); | ||||
|     } | ||||
|  | ||||
|     cnt = 0; | ||||
|     while (++cnt < cntLimit && | ||||
|         m_worker->mainWindow()->updateStep() != UpdateDcEvent::UpdateStep::BL_IS_UP) { | ||||
|         // wait until all bl_startBL() are done | ||||
|         QThread::msleep(200); | ||||
|     } | ||||
|  | ||||
|     if (cnt == cntLimit) { | ||||
|         // really not up | ||||
|         Utils::printCriticalErrorMsg("BL_IS_UP EVENT NOT RECEIVED AFTER 20 SECS"); | ||||
|         stopBootloader(); // try to stop bootloader whichhas been already started | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     if (m_worker->mainWindow()->updateStep() == UpdateDcEvent::UpdateStep::BL_IS_UP) { | ||||
|         // bootloader MUST be running to download device-controller | ||||
| #if 0 | ||||
|         if (!downloadBinaryToDC(bFile)) { | ||||
|             Utils::printCriticalErrorMsg( | ||||
|                 QString("UPDATING DC: ") + bFile + " ...DOWNLOAD FAILED"); | ||||
|         } | ||||
| #endif | ||||
|  | ||||
|     } else { | ||||
|         Utils::printCriticalErrorMsg( | ||||
|             QString("UPDATING DC: ") + bFile + " BOOT LOADER NOT RUNNING -> NO DOWNLOAD (" | ||||
|                   + QThread::currentThread()->objectName() + ")"); | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     // do this unconditionally, even if bootloader is not running at all -> | ||||
|     // the controller possibly tells us nonsense. | ||||
|     if (!stopBootloader()) { | ||||
|         Utils::printCriticalErrorMsg( | ||||
|             QString("UPDATING DC: ") + bFile + " BOOT LOADER STILL RUNNING (" | ||||
|                   + QThread::currentThread()->objectName() + ")"); | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     Utils::printInfoMsg(QString("UPDATING DC: ") + bFile + " ...OK"); | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| QString Update::jsonType(enum FileTypeJson type) { | ||||
|     switch (type) { | ||||
|     case FileTypeJson::CASH: return "CASH"; | ||||
|     case FileTypeJson::CONFIG: return "CONFIG"; | ||||
|     case FileTypeJson::PRINTER: return "PRINTER"; | ||||
|     case FileTypeJson::SERIAL: return "SERIAL"; | ||||
|     case FileTypeJson::DEVICE: return "DEVICE"; | ||||
|     case FileTypeJson::TIME: return "TIME"; | ||||
|     } | ||||
|     return "N/A"; | ||||
| } | ||||
|  | ||||
| bool Update::downloadJson(enum FileTypeJson type, | ||||
|                           int templateIdx, | ||||
|                           QString jsFileToSendToDC) const { | ||||
|  | ||||
|  | ||||
|     Utils::printInfoMsg( | ||||
|         QString("UPDATING JSON-FILE=%1, TEMPLATE-INDEX=%2, JSON-TYPE=%3") | ||||
|             .arg(jsFileToSendToDC) | ||||
|             .arg(templateIdx) | ||||
|             .arg(jsonType(type))); | ||||
|  | ||||
|     m_hw->dc_autoRequest(true); // downloading Json needs the AutoEmission flag | ||||
|     qDebug() << "SET AUTO-REQUEST=TRUE"; | ||||
|     QThread::sleep(1);  // make sure the auto-request flag is acknowledged | ||||
|  | ||||
|     bool ready = false; | ||||
|     int nTry = 25; | ||||
|     while ((ready = m_hw->sys_ready4sending()) == false) { | ||||
|         QThread::msleep(200); | ||||
|         if (--nTry <= 0) { | ||||
|             Utils::printCriticalErrorMsg("SYS NOT READY FOR SENDING AFTER 5 SECONDS"); | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     bool ret = false; | ||||
|     if (ready) { | ||||
|         QFile file(jsFileToSendToDC); | ||||
|         QFileInfo fi(jsFileToSendToDC); // max. size of template file is 800 bytes | ||||
|         if (file.exists()) { | ||||
|             if (file.open(QIODevice::ReadOnly)) { | ||||
|                 if (fi.size() <= 800) { | ||||
|                     QByteArray ba = file.readAll(); | ||||
|                     // kindOfFile: 1=config, 2=device, 3=cash, 4=serial, 5=time, 6=printer | ||||
|                     //      nrOfTemplate=1...32 if kindOfFile==6 | ||||
|                     //      content = content of the Json file, max 800byte ascii signs | ||||
|                     if (m_hw->sys_sendJsonFileToDc((uint8_t)(type), | ||||
|                                                    templateIdx, | ||||
|                                                    (uint8_t *)ba.data())) { | ||||
|                         QThread::sleep(1); | ||||
|                         ret = true; | ||||
|                     } | ||||
|                 } else { | ||||
|                     Utils::printCriticalErrorMsg( | ||||
|                         QString("SIZE OF %1 TOO BIG (%2 BYTES)") | ||||
|                             .arg(jsFileToSendToDC).arg(fi.size())); | ||||
|                 } | ||||
|             } else { | ||||
|                 Utils::printCriticalErrorMsg( | ||||
|                     QString("CAN NOT OPEN ") + jsFileToSendToDC + " FOR READING"); | ||||
|             } | ||||
|         } else { | ||||
|             Utils::printCriticalErrorMsg( | ||||
|                 QString(jsFileToSendToDC) + " DOES NOT EXIST"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     m_hw->dc_autoRequest(false); | ||||
|     qDebug() << "SET AUTO-REQUEST=FALSE"; | ||||
|     QThread::sleep(1);  // make sure the auto-request flag is acknowledged | ||||
|  | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| bool Update::updatePrinterTemplate(int templateIdx, QString jsFile) const { | ||||
|     return downloadJson(FileTypeJson::PRINTER, templateIdx, jsFile); | ||||
| } | ||||
|  | ||||
| bool Update::updateConfig(QString jsFile) { | ||||
|     return downloadJson(FileTypeJson::CONFIG, 0, jsFile); | ||||
| } | ||||
|  | ||||
| bool Update::updateCashConf(QString jsFile) { | ||||
|     return downloadJson(FileTypeJson::CASH, 0, jsFile); | ||||
| } | ||||
|  | ||||
| bool Update::updateDeviceConf(QString jsFile) { | ||||
|     return downloadJson(FileTypeJson::DEVICE, 0, jsFile); | ||||
| } | ||||
|  | ||||
| QStringList Update::split(QString line, QChar sep) { | ||||
|     QStringList lst; | ||||
|     QString next; | ||||
|     int start = 0, end; | ||||
|  | ||||
|     while ((end = line.indexOf(sep, start)) != -1) { | ||||
|         next = line.mid(start, end - start).trimmed(); | ||||
|         lst << next; | ||||
|         start = end + 1; | ||||
|     } | ||||
|     next = line.mid(start, end - start).trimmed(); | ||||
|     lst << next; | ||||
|  | ||||
|     return lst; | ||||
| } | ||||
|  | ||||
| void Update::readyReadStandardOutput() { | ||||
|     QProcess *p = (QProcess *)sender(); | ||||
|     QByteArray buf = p->readAllStandardOutput(); | ||||
|     qCritical() << buf; | ||||
| } | ||||
|  | ||||
| void Update::readyReadStandardError() { | ||||
|     QProcess *p = (QProcess *)sender(); | ||||
|     QByteArray buf = p->readAllStandardError(); | ||||
|     qCritical() << buf; | ||||
| } | ||||
|  | ||||
| void Update::finished(int /*exitCode*/, QProcess::ExitStatus /*exitStatus*/) { | ||||
|     QProcess *p = (QProcess *)sender(); | ||||
|     disconnect(p, SIGNAL(finished(int,QProcess::ExitStatus)), this, SLOT(readyReadStandardOutput())); | ||||
|     disconnect(p, SIGNAL(finished(int,QProcess::ExitStatus)), this, SLOT(readyReadStandardError())); | ||||
| } | ||||
|  | ||||
| QStringList Update::getDcSoftAndHardWareVersion() { | ||||
|     m_hw->dc_autoRequest(true); | ||||
|     QThread::sleep(1); // make sure the timer-slots are active | ||||
|  | ||||
|     for (int i=0; i < 3; ++i) { // send explicit reuests to get | ||||
|                                 // current SW/HW-versions | ||||
|         m_hw->request_DC2_SWversion(); | ||||
|         m_hw->request_DC2_HWversion(); | ||||
|         QThread::sleep(1); | ||||
|     } | ||||
|  | ||||
|     QString const &hwVersion = m_hw->dc_getHWversion().toLower().trimmed(); | ||||
|     QString const &swVersion = m_hw->dc_getSWversion().toLower().trimmed(); | ||||
|  | ||||
|     m_hw->dc_autoRequest(false); | ||||
|     QThread::sleep(1); // make sure the timer-slots are inactive | ||||
|  | ||||
|     if (!hwVersion.isEmpty() && !swVersion.isEmpty()) { | ||||
|         return QStringList() << hwVersion << swVersion; | ||||
|     } | ||||
|  | ||||
|     return QStringList() << "DC HW-version not available" | ||||
|                          << "DC SW-version not available"; | ||||
| } | ||||
|  | ||||
| bool Update::doUpdate(int &displayIndex, QStringList const &filesToWorkOn) { | ||||
|     int tries = 20; | ||||
|     while (!m_hw->sys_areDCdataValid()) { // must deliver 'true', only then are all | ||||
|                                           // data from hwapi valid | ||||
|         if (--tries < 0) { | ||||
|             qCritical() << "ERROR!!! DC DATA NOT VALID -> CA-MASTER-PLUGIN NOT CONNECTED"; | ||||
|             return false; | ||||
|         } | ||||
|         m_hw->dc_autoRequest(true); | ||||
|         QThread::msleep(500); | ||||
|     } | ||||
|  | ||||
|     bool res = false; | ||||
|     QList<QString>::const_iterator it; | ||||
|     for (it = filesToWorkOn.cbegin(); it != filesToWorkOn.cend(); ++it) { | ||||
|         m_worker->startProgressLoop(); | ||||
|         QString const &fToWorkOn = QDir::cleanPath(m_customerRepository + QDir::separator() + it->trimmed()); | ||||
| #if UPDATE_DC == 1 | ||||
|         static const QRegularExpression version("^.*dc2c[.][0-9]{1,2}[.][0-9]{1,2}[.]bin.*$"); | ||||
|         if (fToWorkOn.contains(version)) { | ||||
|             Utils::printInfoMsg("DO-UPDATE FILE-TO-WORK-ON " + fToWorkOn); | ||||
|  | ||||
|             QFile fn(fToWorkOn); | ||||
|             QFileInfo finfo(fn); | ||||
|             if (!fn.exists()) { // check for broken link | ||||
|                 Utils::printCriticalErrorMsg("DO-UPDATE FILE-TO-WORK-ON " | ||||
|                     + fToWorkOn + " DOES NOT EXIST"); | ||||
|                 res = false; | ||||
|             } else { | ||||
|                 bool updateBinaryRes = true; | ||||
|  | ||||
|                 qInfo() << "DOWNLOADING" << finfo.completeBaseName() << "TO DC"; | ||||
|                 m_hw->dc_autoRequest(false);// default: turn auto-request setting off | ||||
|                 QThread::sleep(1);          // wait to be sure that there are no more | ||||
|                                             // commands sent to dc-hardware | ||||
|                 qInfo() << "SET AUTO-REQUEST=FALSE"; | ||||
|  | ||||
|                 if ((updateBinaryRes = updateBinary(fToWorkOn.toStdString().c_str())) == true) { | ||||
|                     qCritical() << "downloaded binary" << fToWorkOn; | ||||
|                     ++displayIndex; | ||||
|                     emit m_worker->appendText(QString("\n(") + QString("%1").arg(displayIndex).rightJustified(2, ' ') + QString(")") | ||||
|                         + QString(" Update ") + QFileInfo(fToWorkOn).fileName(), | ||||
|                         Worker::UPDATE_STEP_DONE); | ||||
|                 } | ||||
|  | ||||
|                 m_hw->dc_autoRequest(true); // turn auto-request setting on | ||||
|                 qInfo() << "SET AUTO-REQUEST=TRUE"; | ||||
|  | ||||
|                 QStringList const &versions = Update::getDcSoftAndHardWareVersion(); | ||||
|                 if (versions.size() >= 2) { | ||||
|                     if (updateBinaryRes == true) { | ||||
|                         qInfo() << "dc-hardware-version (UPDATED)" << versions[0]; | ||||
|                         qInfo() << "dc-firmware-version (UPDATED)" << versions[1]; | ||||
|                     } else { | ||||
|                         qInfo() << "dc-hardware-version (NOT UPDATED)" << versions[0]; | ||||
|                         qInfo() << "dc-firmware-version (NOT UPDATED)" << versions[1]; | ||||
|                     } | ||||
|                 } | ||||
|                 res = updateBinaryRes; | ||||
|             } | ||||
| #endif | ||||
|         if (fToWorkOn.contains("DC2C_print", Qt::CaseInsensitive) | ||||
|          && fToWorkOn.endsWith(".json", Qt::CaseInsensitive)) { | ||||
|             res = true; | ||||
|             int i = fToWorkOn.indexOf("DC2C_print", Qt::CaseInsensitive); | ||||
|             int const templateIdx = fToWorkOn.mid(i).midRef(10, 2).toInt(); | ||||
|             if ((templateIdx < 1) || (templateIdx > 32)) { | ||||
|                 qCritical() << "WRONG TEMPLATE INDEX" << templateIdx; | ||||
|                 res = false; | ||||
|             } else { | ||||
|                 if ((res = updatePrinterTemplate(templateIdx, fToWorkOn))) { | ||||
|                     Utils::printInfoMsg( | ||||
|                         QString("DOWNLOADED PRINTER TEMPLATE %1 WITH INDEX=%2") | ||||
|                             .arg(fToWorkOn) | ||||
|                             .arg(templateIdx)); | ||||
|                     ++displayIndex; | ||||
|                     emit m_worker->appendText(QString("\n(") + QString("%1").arg(displayIndex).rightJustified(3, ' ') + QString(")") | ||||
|                           + QString(" Update ") + QFileInfo(fToWorkOn).fileName(), | ||||
|                             Worker::UPDATE_STEP_DONE); | ||||
|                 } | ||||
|             } | ||||
|         } else if (fToWorkOn.contains("DC2C_cash", Qt::CaseInsensitive) | ||||
|                 && fToWorkOn.endsWith(".json", Qt::CaseInsensitive)) { | ||||
|             res = true; | ||||
|             if ((res = updateCashConf(fToWorkOn))) { | ||||
|                 Utils::printInfoMsg(QString("DOWNLOADED CASH TEMPLATE %1").arg(fToWorkOn)); | ||||
|                 ++displayIndex; | ||||
|                 emit m_worker->appendText(QString("\n(") + QString("%1").arg(displayIndex).rightJustified(3, ' ') + QString(")") | ||||
|                       + QString(" Update ") + QFileInfo(fToWorkOn).fileName(), | ||||
|                         Worker::UPDATE_STEP_DONE); | ||||
|             } | ||||
|         } else if (fToWorkOn.contains("DC2C_conf", Qt::CaseInsensitive) | ||||
|                 && fToWorkOn.endsWith(".json", Qt::CaseInsensitive)) { | ||||
|             res = true; | ||||
|             if ((res= updateConfig(fToWorkOn))) { | ||||
|                 Utils::printInfoMsg(QString("DOWNLOADED CONFIG TEMPLATE %1").arg(fToWorkOn)); | ||||
|                 ++displayIndex; | ||||
|                 emit m_worker->appendText(QString("\n(") + QString("%1").arg(displayIndex).rightJustified(3, ' ') + QString(")") | ||||
|                       + QString(" Update ") + QFileInfo(fToWorkOn).fileName(), | ||||
|                         Worker::UPDATE_STEP_DONE); | ||||
|             } | ||||
|         } else if (fToWorkOn.contains("DC2C_device", Qt::CaseInsensitive) | ||||
|                 && fToWorkOn.endsWith(".json", Qt::CaseInsensitive)) { | ||||
|             res = true; | ||||
|             if ((res = updateDeviceConf(fToWorkOn))) { | ||||
|                 Utils::printInfoMsg(QString("DOWNLOADED DEVICE TEMPLATE %1").arg(fToWorkOn)); | ||||
|                 ++displayIndex; | ||||
|                 emit m_worker->appendText(QString("\n(") + QString("%1").arg(displayIndex).rightJustified(3, ' ') + QString(")") | ||||
|                       + QString(" Update ") + QFileInfo(fToWorkOn).fileName(), | ||||
|                         Worker::UPDATE_STEP_DONE); | ||||
|             } | ||||
|         } else { | ||||
|             qCritical() << "UNKNOWN JSON FILE NAME" << fToWorkOn; | ||||
|             res = false; | ||||
|         } | ||||
|         // m_worker->stopProgressLoop(); | ||||
|         // m_worker->setProgress(100); | ||||
|  | ||||
|         if (res == false) { | ||||
|             break; | ||||
|         } | ||||
|     } // for (it = openLines.cbegin(); it != openLines.end(); ++it) { | ||||
|  | ||||
|     m_hw->dc_autoRequest(true);  // ALWAYS turn autoRequest ON | ||||
|     qDebug() << "SET AUTO-REQUEST=TRUE"; | ||||
|  | ||||
|     return res; | ||||
| } | ||||
		Reference in New Issue
	
	Block a user