2023-07-10 15:57:59 +02:00
|
|
|
#include "command.h"
|
2024-12-20 13:01:34 +01:00
|
|
|
#include "worker.h"
|
2023-07-10 15:57:59 +02:00
|
|
|
|
|
|
|
#include <QProcess>
|
|
|
|
#include <QDebug>
|
2023-08-16 10:39:46 +02:00
|
|
|
#include <QDir>
|
2023-07-10 15:57:59 +02:00
|
|
|
#include <QRegularExpression>
|
2023-10-31 09:12:43 +01:00
|
|
|
#include <QDateTime>
|
2024-12-06 11:48:21 +01:00
|
|
|
#include <QMutexLocker>
|
2023-07-10 15:57:59 +02:00
|
|
|
|
2024-12-20 13:01:34 +01:00
|
|
|
|
2023-07-10 15:57:59 +02:00
|
|
|
Command::Command(QString const &command, int start_timeout, int finish_timeout)
|
|
|
|
: m_command(command.trimmed())
|
|
|
|
, m_commandResult("")
|
|
|
|
, m_waitForStartTimeout(start_timeout)
|
2023-08-14 14:28:23 +02:00
|
|
|
, m_waitForFinishTimeout(finish_timeout)
|
2024-12-20 13:01:34 +01:00
|
|
|
, m_exitCode(-1)
|
|
|
|
, m_p(nullptr)
|
|
|
|
, m_worker(nullptr) {
|
2023-07-10 15:57:59 +02:00
|
|
|
}
|
|
|
|
|
2024-12-06 11:48:21 +01:00
|
|
|
QString Command::getCommandResult(bool reset) const {
|
|
|
|
QMutexLocker locker(&m_mtx);
|
|
|
|
|
|
|
|
if (reset == false) {
|
|
|
|
return m_commandResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString commandResult = m_commandResult;
|
|
|
|
m_commandResult.clear();
|
|
|
|
|
|
|
|
return commandResult;
|
2023-07-10 15:57:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void Command::readyReadStandardOutput() {
|
2024-12-06 11:48:21 +01:00
|
|
|
QMutexLocker locker(&m_mtx);
|
2023-07-10 15:57:59 +02:00
|
|
|
QProcess *p = (QProcess *)sender();
|
2024-12-20 13:01:34 +01:00
|
|
|
if (p) {
|
|
|
|
QString s = p->readAllStandardOutput();
|
2025-01-09 15:22:10 +01:00
|
|
|
|
2024-12-20 13:01:34 +01:00
|
|
|
if (m_worker) {
|
|
|
|
int i = -1;
|
|
|
|
if ((i = s.indexOf("<DC-VERSION>")) != -1) {
|
|
|
|
s = s.mid(i+12).trimmed();
|
|
|
|
if ((i = s.indexOf("\"")) != -1) {
|
|
|
|
s = s.mid(i+1);
|
|
|
|
if ((i = s.indexOf("\"")) != -1) {
|
|
|
|
s = s.mid(0, i).trimmed();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
emit m_worker->showDcDownload(s);
|
|
|
|
} else
|
2025-01-09 15:22:10 +01:00
|
|
|
if ((i = s.indexOf("<DC-PROGRESS>")) != -1) {
|
2024-12-20 13:01:34 +01:00
|
|
|
bool ok;
|
2025-01-09 15:22:10 +01:00
|
|
|
int v = s.mid(i+13).trimmed().toInt(&ok);
|
2024-12-20 13:01:34 +01:00
|
|
|
if (ok) {
|
|
|
|
emit m_worker->setDcDownloadProgress(v);
|
|
|
|
emit m_worker->insertText(s.mid(0,i) + "\n");
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
if ((i = s.indexOf("<DC-UPDATE-FINISH>")) != -1) {
|
|
|
|
m_worker->summary();
|
|
|
|
// qApp->exit(0);
|
|
|
|
} else
|
|
|
|
if ((i = s.indexOf("<DC-UPDATE-SUCCESS>")) != -1) {
|
|
|
|
m_worker->summary();
|
|
|
|
} else
|
|
|
|
if ((i = s.indexOf("<DC-UPDATE-FAILURE>")) != -1) {
|
|
|
|
m_worker->summary();
|
|
|
|
//qApp->exit(-1);
|
2025-01-09 15:22:10 +01:00
|
|
|
} else
|
|
|
|
if ((i = s.indexOf("<JS-PROGRESS>")) != -1) {
|
|
|
|
bool ok;
|
|
|
|
int v = s.mid(i+13).trimmed().toInt(&ok);
|
|
|
|
if (ok) {
|
|
|
|
emit m_worker->setDcDownloadProgress(v);
|
|
|
|
emit m_worker->insertText(s.mid(0,i) + "\n");
|
|
|
|
}
|
2024-12-20 13:01:34 +01:00
|
|
|
} else {
|
|
|
|
emit m_worker->insertText(s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m_commandResult += s;
|
|
|
|
}
|
2023-07-10 15:57:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void Command::readyReadStandardError() {
|
|
|
|
QProcess *p = (QProcess *)sender();
|
2024-12-20 13:01:34 +01:00
|
|
|
if (p) {
|
|
|
|
QByteArray buf = p->readAllStandardError();
|
|
|
|
qCritical() << buf;
|
|
|
|
}
|
2023-07-10 15:57:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
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()) {
|
2024-12-06 11:48:21 +01:00
|
|
|
QMutexLocker locker(&m_mtx);
|
2023-07-10 15:57:59 +02:00
|
|
|
m_commandResult += d;
|
|
|
|
}
|
|
|
|
disconnect(p, SIGNAL(finished(int,QProcess::ExitStatus)), this, SLOT(readyReadStandardOutput()));
|
|
|
|
disconnect(p, SIGNAL(finished(int,QProcess::ExitStatus)), this, SLOT(readyReadStandardError()));
|
2025-01-09 15:22:10 +01:00
|
|
|
disconnect(p, SIGNAL(finished(int,QProcess::ExitStatus)), this, SLOT(finished(int,QProcess::ExitStatus)));
|
|
|
|
|
|
|
|
if (m_command.contains("ATBDownloadDCJsonFiles")) {
|
|
|
|
m_worker->dcUpdate();
|
|
|
|
}
|
2023-07-10 15:57:59 +02:00
|
|
|
}
|
|
|
|
|
2024-12-20 13:01:34 +01:00
|
|
|
bool Command::start(QString workingDirectory, QStringList args) {
|
|
|
|
if (!QDir::setCurrent(workingDirectory)) {
|
|
|
|
qCritical() << "SET WORKING_DIRECTORY" << workingDirectory
|
|
|
|
<< "FAILED FOR" << m_command;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (m_p != nullptr) {
|
|
|
|
delete m_p;
|
|
|
|
}
|
2025-01-09 15:22:10 +01:00
|
|
|
|
|
|
|
qCritical() << "COMMAND" << m_command << workingDirectory << args;
|
|
|
|
|
2024-12-20 13:01:34 +01:00
|
|
|
m_p = new QProcess(this);
|
|
|
|
m_p->setWorkingDirectory(workingDirectory);
|
|
|
|
m_p->setProcessChannelMode(QProcess::MergedChannels);
|
|
|
|
|
|
|
|
connect(m_p, SIGNAL(readyReadStandardOutput()), this, SLOT(readyReadStandardOutput()));
|
|
|
|
connect(m_p, SIGNAL(readyReadStandardError()), this, SLOT(readyReadStandardError()));
|
2025-01-09 15:22:10 +01:00
|
|
|
connect(m_p, SIGNAL(finished(int,QProcess::ExitStatus)), this, SLOT(finished(int,QProcess::ExitStatus)));
|
2024-12-20 13:01:34 +01:00
|
|
|
|
|
|
|
if (!args.isEmpty()) {
|
|
|
|
m_p->start(m_command, args);
|
|
|
|
} else {
|
|
|
|
m_p->start(m_command);
|
|
|
|
}
|
|
|
|
|
|
|
|
return m_p->waitForStarted(m_waitForStartTimeout);
|
|
|
|
}
|
|
|
|
|
2023-07-14 12:56:33 +02:00
|
|
|
bool Command::execute(QString workingDirectory, QStringList args) {
|
2023-08-16 10:39:46 +02:00
|
|
|
|
|
|
|
if (!QDir::setCurrent(workingDirectory)) {
|
|
|
|
qCritical() << "SET WORKING_DIRECTORY" << workingDirectory
|
|
|
|
<< "FAILED FOR" << m_command;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-07-10 15:57:59 +02:00
|
|
|
QScopedPointer<QProcess> p(new QProcess(this));
|
2023-08-16 10:39:46 +02:00
|
|
|
p->setWorkingDirectory(workingDirectory);
|
2023-07-10 15:57:59 +02:00
|
|
|
p->setProcessChannelMode(QProcess::MergedChannels);
|
|
|
|
|
|
|
|
connect(&(*p), SIGNAL(readyReadStandardOutput()), this, SLOT(readyReadStandardOutput()));
|
|
|
|
connect(&(*p), SIGNAL(readyReadStandardError()), this, SLOT(readyReadStandardError()));
|
|
|
|
|
2023-07-14 12:56:33 +02:00
|
|
|
if (!args.isEmpty()) {
|
2024-05-15 15:03:47 +02:00
|
|
|
//qDebug() << "START COMMAND" << m_command << "WITH ARGS" << args
|
|
|
|
// << "IN" << p->workingDirectory();
|
2023-07-14 12:56:33 +02:00
|
|
|
p->start(m_command, args);
|
|
|
|
} else {
|
2024-05-15 15:03:47 +02:00
|
|
|
//qDebug() << "START COMMAND" << m_command
|
|
|
|
// << "IN" << p->workingDirectory();
|
2023-07-14 12:56:33 +02:00
|
|
|
p->start(m_command);
|
|
|
|
}
|
2023-07-10 15:57:59 +02:00
|
|
|
|
2023-10-31 09:12:43 +01:00
|
|
|
qint64 const start = QDateTime::currentDateTime().toMSecsSinceEpoch();
|
|
|
|
|
2023-07-10 15:57:59 +02:00
|
|
|
if (p->waitForStarted(m_waitForStartTimeout)) {
|
2024-05-15 15:03:47 +02:00
|
|
|
// qDebug() << "PROCESS" << m_command << "STARTED IN" << p->workingDirectory();
|
2023-07-10 15:57:59 +02:00
|
|
|
if (p->state() == QProcess::ProcessState::Running) {
|
2024-05-15 15:03:47 +02:00
|
|
|
// qDebug() << "PROCESS" << m_command << "RUNNING IN" << p->workingDirectory();
|
2023-10-10 16:03:52 +02:00
|
|
|
// 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;
|
|
|
|
}
|
2023-10-05 10:54:19 +02:00
|
|
|
bool const no_timeout = p->waitForFinished(wait);
|
|
|
|
if (no_timeout) {
|
2024-05-15 15:03:47 +02:00
|
|
|
// qDebug() << "PROCESS" << m_command << "FINISHED IN" << p->workingDirectory();
|
2023-07-10 15:57:59 +02:00
|
|
|
if (p->exitStatus() == QProcess::NormalExit) {
|
2023-08-14 14:28:23 +02:00
|
|
|
if ((m_exitCode = p->exitCode()) == 0) {
|
2023-10-31 09:12:43 +01:00
|
|
|
qint64 const end = QDateTime::currentDateTime().toMSecsSinceEpoch();
|
2023-08-16 10:39:46 +02:00
|
|
|
qDebug() << "EXECUTED" << m_command
|
2023-10-31 09:12:43 +01:00
|
|
|
<< QString("(runtime %1ms)").arg(end-start)
|
2023-08-16 10:39:46 +02:00
|
|
|
<< "with code" << m_exitCode
|
|
|
|
<< "IN" << p->workingDirectory();
|
2023-08-04 13:31:12 +02:00
|
|
|
return true;
|
|
|
|
} else {
|
2023-10-31 09:12:43 +01:00
|
|
|
qint64 const end = QDateTime::currentDateTime().toMSecsSinceEpoch();
|
2023-08-16 10:39:46 +02:00
|
|
|
qCritical() << "EXECUTED" << m_command
|
2023-10-31 09:12:43 +01:00
|
|
|
<< QString("(runtime %1ms)").arg(end-start)
|
2023-08-16 10:39:46 +02:00
|
|
|
<< "with code" << m_exitCode
|
|
|
|
<< "IN" << p->workingDirectory();
|
2023-08-04 13:31:12 +02:00
|
|
|
}
|
2023-07-10 15:57:59 +02:00
|
|
|
} else {
|
2023-10-31 09:12:43 +01:00
|
|
|
qint64 const end = QDateTime::currentDateTime().toMSecsSinceEpoch();
|
2023-07-10 15:57:59 +02:00
|
|
|
qCritical() << "PROCESS" << m_command << "CRASHED with code"
|
2023-08-16 10:39:46 +02:00
|
|
|
<< p->exitCode()
|
2023-10-31 09:12:43 +01:00
|
|
|
<< QString("(after %1ms)").arg(end-start)
|
2023-08-16 10:39:46 +02:00
|
|
|
<< "IN" << p->workingDirectory();
|
2023-07-10 15:57:59 +02:00
|
|
|
}
|
|
|
|
} else {
|
2023-10-31 09:12:43 +01:00
|
|
|
qint64 const end = QDateTime::currentDateTime().toMSecsSinceEpoch();
|
2023-08-16 10:39:46 +02:00
|
|
|
qCritical() << "PROCESS" << m_command
|
2023-10-05 10:54:19 +02:00
|
|
|
<< "DID NOT FINISH WITH" << wait
|
2023-10-31 09:12:43 +01:00
|
|
|
<< "MS IN" << p->workingDirectory()
|
|
|
|
<< QString("(runtime %1ms)").arg(end-start);
|
2023-07-10 15:57:59 +02:00
|
|
|
}
|
|
|
|
} else {
|
2023-08-16 10:39:46 +02:00
|
|
|
qCritical() << "WRONG PROCESS STATE" << p->state()
|
|
|
|
<< "IN" << p->workingDirectory();
|
2023-07-10 15:57:59 +02:00
|
|
|
}
|
|
|
|
} else {
|
2023-10-31 09:12:43 +01:00
|
|
|
qint64 const end = QDateTime::currentDateTime().toMSecsSinceEpoch();
|
2023-08-16 10:39:46 +02:00
|
|
|
qCritical() << "PROCESS" << m_command << "TIMEOUT AT START"
|
2023-10-31 09:12:43 +01:00
|
|
|
<< QString("(runtime %1ms)").arg(end-start)
|
2023-08-16 10:39:46 +02:00
|
|
|
<< "IN" << p->workingDirectory();
|
2023-07-10 15:57:59 +02:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|