mainwindow.cpp 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492
  1. #include <include/ui/mainwindow.h>
  2. #include <QDebug>
  3. #include <QLayout>
  4. #include <QScrollBar>
  5. #include <QString>
  6. #include <QTextEdit>
  7. #include <QTextStream>
  8. #include <headers/util.hpp>
  9. #include <vector>
  10. #include "ui_mainwindow.h"
  11. /**
  12. * Helper functions
  13. */
  14. namespace {
  15. bool isSameEvent(QString event1, QString event2) {
  16. auto event_size =
  17. event1.size() > event2.size() ? event2.size() : event1.size();
  18. auto similarity_standard = event_size * 0.67;
  19. auto similarity_index = 0;
  20. for (auto i = 0; i < event_size; i++) {
  21. if (event1[i] == event2[i]) {
  22. similarity_index++;
  23. }
  24. }
  25. return similarity_index > similarity_standard;
  26. }
  27. int getLikeEventNum(QString event, QList<QString> events) {
  28. auto i = events.size() - 1;
  29. auto hits = 0;
  30. bool is_same_event = false;
  31. auto incoming_event = event.remove(0, 11);
  32. do {
  33. auto existing_event = events[i];
  34. if ((is_same_event =
  35. isSameEvent(incoming_event, existing_event.remove(0, 11)))) {
  36. i--;
  37. hits++;
  38. }
  39. } while (is_same_event && i > -1);
  40. return hits;
  41. }
  42. QString timestampPrefix() {
  43. return TimeUtils::getTime() + " - ";
  44. }
  45. /**
  46. * @brief createProcessListItem
  47. * @param process
  48. * @return
  49. */
  50. QStandardItem* createProcessListItem(Process process) {
  51. return new QStandardItem(
  52. QString("%0 requested for execution. ID: %1\nStatus: %2\nTime: %3 "
  53. "Done: %4\n Errors: %5")
  54. .arg(process.name)
  55. .arg(process.id)
  56. .arg(ProcessNames[process.state - 1])
  57. .arg(process.start)
  58. .arg(process.end)
  59. .arg(process.error));
  60. }
  61. /**
  62. * @brief createEventListItem
  63. * @param event
  64. * @return
  65. */
  66. QStandardItem* createEventListItem(QString event) {
  67. return new QStandardItem(event);
  68. }
  69. } // namespace
  70. /**
  71. *\mainpage The KYGUI application interface begins with the MainWindow
  72. * @brief MainWindow::MainWindow
  73. * @param argc
  74. * @param argv
  75. * @param parent
  76. */
  77. MainWindow::MainWindow(int argc, char** argv, QWidget* parent)
  78. : QMainWindow(parent),
  79. cli_argc(argc),
  80. cli_argv(argv),
  81. ui(new Ui::MainWindow),
  82. arg_ui(new ArgDialog),
  83. q_client(nullptr) {
  84. m_event_model = new QStandardItemModel(this);
  85. m_process_model = new QStandardItemModel(this);
  86. q_client = new Client(this, cli_argc, cli_argv);
  87. message_parser.init(this);
  88. ui->setupUi(this);
  89. this->setWindowTitle("KYGUI");
  90. setStyleSheet(
  91. "QListView { font: 87 11pt \"Noto Sans\"; background-color: #2f535f;"
  92. "alternate-background-color: #616161; color: rgb(131, 148, 150); "
  93. "font-weight: 700; background-color: rgb(29, 51, 59);"
  94. "color: rgb(223, 252, 255);}");
  95. setConnectScreen();
  96. connect(ui->connect, &QPushButton::clicked, this, &MainWindow::connectClient);
  97. ui->eventList->setModel(m_event_model);
  98. ui->processList->setModel(m_process_model);
  99. }
  100. /**
  101. * @brief MainWindow::~MainWindow
  102. */
  103. MainWindow::~MainWindow() {
  104. delete q_client;
  105. delete ui;
  106. delete arg_ui;
  107. delete m_event_model;
  108. }
  109. /**
  110. * @brief MainWindow::setConnectScreen
  111. * @param visible
  112. */
  113. void MainWindow::setConnectScreen(bool visible) {
  114. if (visible) {
  115. ui->startScreen->activateWindow();
  116. ui->startScreen->raise();
  117. ui->kyConfig->activateWindow();
  118. ui->kyConfig->raise();
  119. ui->startScreen->setMaximumSize(1080, 675);
  120. ui->startScreen->setMinimumSize(1080, 675);
  121. ui->connect->setMaximumSize(1080, 500);
  122. ui->connect->setMinimumSize(1080, 500);
  123. ui->kyConfig->setMaximumSize(1080, 175);
  124. ui->kyConfig->setMinimumSize(1080, 175);
  125. ui->logo->raise();
  126. QFile file(QCoreApplication::applicationDirPath() + "/config/config.json");
  127. file.open(QIODevice::ReadOnly | QFile::ReadOnly);
  128. QString config_json = QString::fromUtf8(file.readAll());
  129. ui->kyConfig->setText(config_json);
  130. qDebug() << "Set config json: \n" << ui->kyConfig->toPlainText();
  131. file.close();
  132. } else {
  133. ui->connect->hide();
  134. ui->kyConfig->hide();
  135. ui->startScreen->setVisible(false);
  136. }
  137. }
  138. /**
  139. * @brief MainWindow::connectClient
  140. */
  141. void MainWindow::connectClient() {
  142. auto text = ui->kyConfig->toPlainText();
  143. qDebug() << text;
  144. m_config = getConfigObject(ui->kyConfig->toPlainText());
  145. QString file_path = m_config.at("fileDirectory");
  146. if (file_path != NULL) {
  147. arg_ui->setFilePath(file_path);
  148. }
  149. setConnectScreen(false);
  150. qDebug() << "Connecting to KServer";
  151. QObject::connect(q_client, &Client::messageReceived, this,
  152. &MainWindow::onMessageReceived);
  153. QProgressBar* progressBar = ui->progressBar;
  154. q_client->start();
  155. for (int i = 1; i < 101; i++) {
  156. progressBar->setValue(i);
  157. }
  158. QPushButton* send_message_button =
  159. this->findChild<QPushButton*>("sendMessage");
  160. // Handle mouse
  161. QObject::connect(send_message_button, &QPushButton::clicked, this, [this]() {
  162. q_client->sendMessage(escapeMessage(ui->inputText->toPlainText()));
  163. ui->inputText->clear();
  164. });
  165. QObject::connect(
  166. ui->appList,
  167. static_cast<void (QComboBox::*)(int)>(&QComboBox::currentIndexChanged),
  168. this, [this]() {
  169. QString app_name = ui->appList->currentText();
  170. q_client->setSelectedApp(std::vector<QString>{{app_name}});
  171. arg_ui->setAppName(app_name);
  172. arg_ui->setConfig(configValue(app_name, m_config, true));
  173. });
  174. QPushButton* disconnect_button = this->findChild<QPushButton*>("disconnect");
  175. QObject::connect(disconnect_button, &QPushButton::clicked, this, [this]() {
  176. q_client->closeConnection();
  177. QApplication::exit(9);
  178. });
  179. QObject::connect(ui->execute, &QPushButton::clicked, this,
  180. [this]() { q_client->execute(); });
  181. QObject::connect(ui->addArgs, &QPushButton::clicked, this, [this]() {
  182. if (ui->appList->count() == 0) {
  183. QMessageBox::warning(this, tr("Args"),
  184. tr("Please connect to the KServer and retrieve a "
  185. "list of available processes."));
  186. } else {
  187. arg_ui->show();
  188. }
  189. });
  190. QObject::connect(ui->openMessages, &QPushButton::clicked, this, [this]() {
  191. message_ui.show();
  192. });
  193. QObject::connect(
  194. arg_ui, &ArgDialog::taskRequestReady, this,
  195. [this](Task* task) {
  196. auto mask = q_client->getSelectedApp();
  197. if (mask > -1) {
  198. q_client->scheduleTask(task);
  199. }
  200. });
  201. QObject::connect(
  202. ui->processList, &QListView::clicked, this,
  203. [this](const QModelIndex& index) {
  204. auto process = m_processes.at(index.row());
  205. QString process_info_text =
  206. m_processes.at(index.row()).name.toUtf8() + "\n";
  207. process_info_text += "Execution requested at " +
  208. process.start.toUtf8() + "\n" +
  209. "Is currently in a state of: " +
  210. ProcessNames[process.state - 1].toUtf8();
  211. if (process.end.size() > 0 || process.id == "Scheduled task") {
  212. process_info_text += "\n\nResult: \n" + process.result;
  213. }
  214. UI::infoMessageBox(process_info_text, "Process");
  215. });
  216. QObject::connect(ui->eventList, &QListView::clicked, this,
  217. [this](const QModelIndex& index) {
  218. UI::infoMessageBox(m_event_model->item(index.row(), index.column())->text(), "Event");
  219. });
  220. QObject::connect(m_event_model, &QAbstractItemModel::rowsAboutToBeInserted,
  221. this, [this]() {
  222. QScrollBar* bar = ui->eventList->verticalScrollBar();
  223. if (bar->value() == bar->maximum()) {
  224. m_view_states.eventViewBottom = true;
  225. }
  226. });
  227. QObject::connect(m_event_model, &QAbstractItemModel::rowsInserted, this,
  228. [this]() {
  229. if (m_view_states.eventViewBottom) {
  230. ui->eventList->scrollToBottom();
  231. }
  232. });
  233. QTimer* timer = new QTimer(this);
  234. connect(timer, &QTimer::timeout, q_client, &Client::ping);
  235. timer->start(10000);
  236. }
  237. /**
  238. * @brief MainWindow::onMessageReceived
  239. * @param t
  240. * @param message
  241. * @param v
  242. */
  243. void MainWindow::onMessageReceived(int t, const QString& message, StringVec v) {
  244. QString timestamp_prefix = timestampPrefix();
  245. if (t == MESSAGE_UPDATE_TYPE) { // Normal message
  246. qDebug() << "Updating message area";
  247. message_parser.handleMessage(message, v);
  248. message_ui.append(message);
  249. } else if (t == COMMANDS_UPDATE_TYPE) { // Received app list from server
  250. qDebug() << "Updating commands";
  251. QString default_app = configValue("defaultApp", m_config);
  252. message_parser.handleCommands(v, default_app);
  253. if (message == "New Session") { // Session has started
  254. ui->led->setState(true);
  255. auto app_name = q_client->getAppName(q_client->getSelectedApp());
  256. arg_ui->setConfig(configValue(app_name, m_config, true));
  257. if (configBoolValue("schedulerMode", std::ref(m_config))) {
  258. arg_ui->show();
  259. }
  260. }
  261. } else if (t == PROCESS_REQUEST_TYPE) { // Sent execution request to server
  262. qDebug() << "Updating process list";
  263. m_processes.push_back(Process{.name = v.at(1),
  264. .state = ProcessState::PENDING,
  265. .start = TimeUtils::getTime(),
  266. .id = v.at(2)});
  267. int row = 0;
  268. for (const auto& process : m_processes) {
  269. m_process_model->setItem(row, createProcessListItem(process));
  270. row++;
  271. }
  272. } else if (t == EVENT_UPDATE_TYPE) { // Received event from server
  273. QString event_message = message_parser.handleEventMessage(message, v);
  274. if (m_events.size() > 1) { // Group repeating event messages
  275. auto last_event = m_events[m_events.size() - 1];
  276. if (isSameEvent(message, last_event.remove(0, 11))) {
  277. m_consecutive_events++;
  278. auto count = getLikeEventNum(event_message, m_events);
  279. QString clean_event_message =
  280. event_message + " (" + QString::number(count) + ")";
  281. m_events.push_back(event_message);
  282. m_event_model->setItem(m_event_model->rowCount() - 1,
  283. createEventListItem(clean_event_message));
  284. return; // It was not a unique message, we can return
  285. }
  286. m_consecutive_events = 0;
  287. }
  288. if (isKEvent<QString>(message,
  289. Event::TASK_SCHEDULED)) { // Event was scheduled task
  290. event_message += ". Details:\n" + parseTaskInfo(v);
  291. arg_ui->notifyClientSuccess(); // Update ArgDialog accordingly
  292. }
  293. m_events.push_back(event_message);
  294. m_event_model->setItem(m_event_model->rowCount(),
  295. createEventListItem(event_message));
  296. } else {
  297. qDebug() << "Unknown update type. Cannot update UI";
  298. }
  299. }
  300. /**
  301. * @brief MainWindow::parseTaskInfo
  302. * @param v
  303. * @return
  304. */
  305. QString MainWindow::parseTaskInfo(StringVec v) {
  306. QString task_info{};
  307. if (q_client == nullptr) {
  308. qDebug() << "Can't parse when not connected";
  309. return task_info;
  310. }
  311. auto size = v.size();
  312. if (size < 3) {
  313. qDebug() << "Not enough arguments to parse task information";
  314. } else {
  315. auto error = size == 4;
  316. task_info += " UUID - " + v.at(0) + "\n ID - " + v.at(1) + "\n APP - " +
  317. q_client->getAppName(std::stoi(v.at(2).toUtf8().constData())) +
  318. "\n ENV - " + (v.at(3));
  319. if (error) {
  320. task_info += "\n !ERROR! - " + v.at(3);
  321. }
  322. }
  323. return task_info;
  324. }
  325. /**
  326. * MessageParser
  327. *
  328. * \note We use the MessageParser class to do the heavy lifting of parsing
  329. * incoming messages and updating the UI accordingly
  330. */
  331. /**
  332. * @brief MainWindow::MessageParser::init
  333. * @param window
  334. */
  335. void MainWindow::MessageParser::init(MainWindow* window) {
  336. this->window = window;
  337. }
  338. /**
  339. * @brief MainWindow::MessageParser::handleCommands
  340. * @param commands
  341. * @param default_command
  342. */
  343. void MainWindow::MessageParser::handleCommands(StringVec commands,
  344. QString default_command) {
  345. QComboBox* app_list = window->ui->appList;
  346. app_list->clear();
  347. int app_index = 0;
  348. for (const auto& s : commands) {
  349. app_list->addItem(s);
  350. if (s.toLower() == default_command.toLower()) {
  351. window->ui->appList->setCurrentIndex(app_index);
  352. std::vector<QString> selected{std::move(default_command)};
  353. window->q_client->setSelectedApp(std::move(selected));
  354. }
  355. app_index++;
  356. }
  357. }
  358. /**
  359. * @brief MainWindow::MessageParser::handleMessage
  360. * @param message
  361. * @param v
  362. */
  363. void MainWindow::MessageParser::handleMessage(QString message, StringVec v) {
  364. window->message_ui.append(timestampPrefix() + parseMessage(message, v), true);
  365. }
  366. /**
  367. * @brief MainWindow::MessageParser::parseMessage
  368. * @param message
  369. * @param v
  370. * @return
  371. */
  372. QString MainWindow::MessageParser::parseMessage(const QString& message,
  373. StringVec v) {
  374. QString simplified_message{};
  375. if (isMessage(message.toUtf8())) {
  376. simplified_message += "Message: " + getMessage(message.toUtf8());
  377. } else if (isEvent(message.toUtf8())) {
  378. simplified_message += "Event: " + getEvent(message.toUtf8());
  379. } else if (isOperation(message.toUtf8())) {
  380. simplified_message += "Operation: ";
  381. simplified_message += getOperation(message.toUtf8()).c_str();
  382. }
  383. return simplified_message;
  384. }
  385. /**
  386. * @brief MainWindow::MessageParser::updateProcessResult
  387. * @param id
  388. * @param result
  389. * @param error
  390. */
  391. void MainWindow::MessageParser::updateProcessResult(
  392. QString id, QString result,
  393. bool error = false) { // We need to start matching processes with a
  394. // unique identifier
  395. for (int i = window->m_processes.size() - 1; i >= 0; i--) {
  396. if (window->m_processes.at(i).id == id) {
  397. window->m_processes.at(i).end = TimeUtils::getTime();
  398. window->m_processes.at(i).state =
  399. !error ? ProcessState::SUCCEEDED : ProcessState::FAILED;
  400. window->m_processes.at(i).result = result;
  401. window->m_process_model->setItem(
  402. i, 0, createProcessListItem(window->m_processes.at(i)));
  403. return;
  404. }
  405. }
  406. // If we didn't return, it's a new process:
  407. }
  408. /**
  409. * @brief MainWindow::MessageParser::handleEventMessage
  410. * @param message
  411. * @param v
  412. * @return
  413. */
  414. QString MainWindow::MessageParser::handleEventMessage(QString message,
  415. StringVec v) {
  416. QString event_message = timestampPrefix();
  417. if (!v.empty()) {
  418. if (v.size() == 1) {
  419. event_message += message + "\n" + v.at(0);
  420. } else {
  421. event_message += message;
  422. if (message == "Process Result") {
  423. auto error = v.size() > 3 ? true : false;
  424. event_message += "\n";
  425. auto app_name = window->q_client->getAppName(
  426. std::stoi(v.at(0).toUtf8().constData()));
  427. auto process_it = std::find_if(
  428. window->m_processes.begin(), window->m_processes.end(),
  429. [v](const Process& process) { return process.id == v.at(1); });
  430. if (process_it != window->m_processes.end()) {
  431. updateProcessResult(v.at(1), v.at(2), error);
  432. } else { // new process, from scheduled task
  433. Process new_process{
  434. .name = app_name,
  435. .state = !error ? ProcessState::SUCCEEDED : ProcessState::FAILED,
  436. .start = TimeUtils::getTime(),
  437. .id = "Scheduled task",
  438. .error = error ? v.at(3) : "No errors reported"};
  439. if (v.count() > 2 && !v.at(2).isEmpty()) {
  440. new_process.result = v.at(2);
  441. new_process.end = new_process.start;
  442. }
  443. window->m_processes.push_back(new_process);
  444. window->m_process_model->setItem(window->m_process_model->rowCount(),
  445. createProcessListItem(new_process));
  446. }
  447. event_message += app_name;
  448. event_message += ": ";
  449. event_message += v.at(2);
  450. if (error) {
  451. event_message += "\n Error: " + v.at(3);
  452. }
  453. } else if (QString::compare(message, "Message Received") == 0) {
  454. event_message += "\n" + v.at(1) + ": " + v.at(2);
  455. }
  456. }
  457. } else {
  458. event_message += message;
  459. }
  460. return event_message;
  461. }