minikube/gui/window.cpp

712 lines
23 KiB
C++
Raw Normal View History

2022-02-24 23:02:53 +00:00
/****************************************************************************
**
2022-03-02 21:14:10 +00:00
** Copyright 2022 The Kubernetes Authors All rights reserved.
**
** Copyright (C) 2021 Anders F Björklund
2022-02-24 23:02:53 +00:00
** Copyright (C) 2016 The Qt Company Ltd.
** Contact: https://www.qt.io/licensing/
**
** This file is part of the examples of the Qt Toolkit.
**
** $QT_BEGIN_LICENSE:BSD$
** Commercial License Usage
** Licensees holding valid commercial Qt licenses may use this file in
** accordance with the commercial license agreement provided with the
** Software or, alternatively, in accordance with the terms contained in
** a written agreement between you and The Qt Company. For licensing terms
** and conditions see https://www.qt.io/terms-conditions. For further
** information use the contact form at https://www.qt.io/contact-us.
**
** BSD License Usage
** Alternatively, you may use this file under the terms of the BSD license
** as follows:
**
** "Redistribution and use in source and binary forms, with or without
** modification, are permitted provided that the following conditions are
** met:
** * Redistributions of source code must retain the above copyright
** notice, this list of conditions and the following disclaimer.
** * Redistributions in binary form must reproduce the above copyright
** notice, this list of conditions and the following disclaimer in
** the documentation and/or other materials provided with the
** distribution.
** * Neither the name of The Qt Company Ltd nor the names of its
** contributors may be used to endorse or promote products derived
** from this software without specific prior written permission.
**
**
** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE."
**
** $QT_END_LICENSE$
**
****************************************************************************/
#include "window.h"
#ifndef QT_NO_SYSTEMTRAYICON
#include <QAction>
#include <QCheckBox>
#include <QComboBox>
#include <QCoreApplication>
#include <QCloseEvent>
#include <QGroupBox>
#include <QLabel>
#include <QLineEdit>
#include <QMenu>
#include <QPushButton>
#include <QSpinBox>
#include <QTextEdit>
#include <QVBoxLayout>
#include <QMessageBox>
#include <QProcess>
#include <QDebug>
#include <QJsonDocument>
#include <QJsonObject>
#include <QJsonArray>
#include <QTableView>
#include <QHeaderView>
#include <QFormLayout>
#include <QDialogButtonBox>
2022-03-02 23:43:14 +00:00
#include <QStandardPaths>
#ifndef QT_NO_TERMWIDGET
#include <QApplication>
#include <QMainWindow>
#include "qtermwidget.h"
#endif
2022-02-24 23:02:53 +00:00
//! [0]
Window::Window()
{
trayIconIcon = new QIcon(":/images/minikube.png");
checkForMinikube();
2022-03-02 21:32:33 +00:00
createClusterGroupBox();
2022-02-24 23:02:53 +00:00
createActions();
createTrayIcon();
2022-03-02 23:43:14 +00:00
connect(sshButton, &QAbstractButton::clicked, this, &Window::sshConsole);
connect(dashboardButton, &QAbstractButton::clicked, this, &Window::dashboardBrowser);
2022-04-06 22:28:18 +00:00
connect(startButton, &QAbstractButton::clicked, this, &Window::startSelectedMinikube);
2022-02-24 23:02:53 +00:00
connect(stopButton, &QAbstractButton::clicked, this, &Window::stopMinikube);
connect(deleteButton, &QAbstractButton::clicked, this, &Window::deleteMinikube);
2022-03-02 21:32:33 +00:00
connect(refreshButton, &QAbstractButton::clicked, this, &Window::updateClusters);
2022-02-24 23:02:53 +00:00
connect(createButton, &QAbstractButton::clicked, this, &Window::initMachine);
connect(trayIcon, &QSystemTrayIcon::messageClicked, this, &Window::messageClicked);
2022-03-02 23:43:14 +00:00
dashboardProcess = 0;
2022-02-24 23:02:53 +00:00
QVBoxLayout *mainLayout = new QVBoxLayout;
2022-03-02 21:32:33 +00:00
mainLayout->addWidget(clusterGroupBox);
2022-02-24 23:02:53 +00:00
setLayout(mainLayout);
trayIcon->show();
setWindowTitle(tr("minikube"));
setWindowIcon(*trayIconIcon);
resize(600, 400);
}
//! [0]
//! [1]
void Window::setVisible(bool visible)
{
minimizeAction->setEnabled(visible);
restoreAction->setEnabled(!visible);
QDialog::setVisible(visible);
}
//! [1]
//! [2]
void Window::closeEvent(QCloseEvent *event)
{
#ifdef Q_OS_OSX
if (!event->spontaneous() || !isVisible()) {
return;
}
#endif
if (trayIcon->isVisible()) {
QMessageBox::information(this, tr("Systray"),
tr("The program will keep running in the "
"system tray. To terminate the program, "
"choose <b>Quit</b> in the context menu "
"of the system tray entry."));
hide();
event->ignore();
}
}
//! [2]
//! [6]
void Window::messageClicked()
{
QMessageBox::information(0, tr("Systray"),
tr("Sorry, I already gave what help I could.\n"
"Maybe you should try asking a human?"));
}
//! [6]
void Window::createActions()
{
minimizeAction = new QAction(tr("Mi&nimize"), this);
connect(minimizeAction, &QAction::triggered, this, &QWidget::hide);
restoreAction = new QAction(tr("&Restore"), this);
connect(restoreAction, &QAction::triggered, this, &QWidget::showNormal);
quitAction = new QAction(tr("&Quit"), this);
connect(quitAction, &QAction::triggered, qApp, &QCoreApplication::quit);
}
2022-03-02 23:43:14 +00:00
static QString minikubePath()
{
QString program = QStandardPaths::findExecutable("minikube");
if (program.isEmpty()) {
QStringList paths = { "/usr/local/bin" };
program = QStandardPaths::findExecutable("minikube", paths);
}
return program;
}
2022-02-24 23:02:53 +00:00
void Window::createTrayIcon()
{
trayIconMenu = new QMenu(this);
trayIconMenu->addAction(minimizeAction);
trayIconMenu->addAction(restoreAction);
trayIconMenu->addSeparator();
trayIconMenu->addAction(quitAction);
trayIcon = new QSystemTrayIcon(this);
trayIcon->setContextMenu(trayIconMenu);
trayIcon->setIcon(*trayIconIcon);
}
2022-04-06 22:28:18 +00:00
void Window::startMinikube(QStringList moreArgs)
2022-02-24 23:02:53 +00:00
{
QString text;
2022-04-06 22:28:18 +00:00
QStringList args = { "start", "-o", "json" };
args << moreArgs;
bool success = sendMinikubeCommand(args, text);
2022-03-02 21:32:33 +00:00
updateClusters();
if (success) {
return;
}
outputFailedStart(text);
2022-02-24 23:02:53 +00:00
}
2022-04-06 22:28:18 +00:00
void Window::startSelectedMinikube()
{
QStringList args = { "-p", selectedCluster() };
return startMinikube(args);
}
2022-02-24 23:02:53 +00:00
void Window::stopMinikube()
{
2022-03-02 23:53:42 +00:00
QStringList args = { "stop", "-p", selectedCluster() };
2022-02-24 23:02:53 +00:00
sendMinikubeCommand(args);
2022-03-02 21:32:33 +00:00
updateClusters();
2022-02-24 23:02:53 +00:00
}
void Window::deleteMinikube()
{
2022-03-02 23:53:42 +00:00
QStringList args = { "delete", "-p", selectedCluster() };
2022-02-24 23:02:53 +00:00
sendMinikubeCommand(args);
2022-03-02 21:32:33 +00:00
updateClusters();
2022-02-24 23:02:53 +00:00
}
2022-03-02 21:32:33 +00:00
void Window::updateClusters()
2022-02-24 23:02:53 +00:00
{
2022-03-02 21:32:33 +00:00
QString cluster = selectedCluster();
clusterModel->setClusters(getClusters());
setSelectedCluster(cluster);
2022-02-24 23:02:53 +00:00
updateButtons();
}
2022-03-02 21:32:33 +00:00
ClusterList Window::getClusters()
2022-02-24 23:02:53 +00:00
{
2022-03-02 21:32:33 +00:00
ClusterList clusters;
2022-03-02 23:53:42 +00:00
QStringList args = { "profile", "list", "-o", "json" };
2022-02-24 23:02:53 +00:00
QString text;
bool success = sendMinikubeCommand(args, text);
QStringList lines;
if (success) {
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
lines = text.split("\n", Qt::SkipEmptyParts);
#else
lines = text.split("\n", QString::SkipEmptyParts);
#endif
}
for (int i = 0; i < lines.size(); i++) {
QString line = lines.at(i);
QJsonParseError error;
QJsonDocument json = QJsonDocument::fromJson(line.toUtf8(), &error);
if (json.isNull()) {
qDebug() << error.errorString();
continue;
}
if (json.isObject()) {
QJsonObject par = json.object();
QJsonArray a = par["valid"].toArray();
for (int j = 0; j < a.size(); j++) {
QJsonObject obj = a[j].toObject();
QString name;
if (obj.contains("Name")) {
name = obj["Name"].toString();
}
if (name.isEmpty()) {
continue;
}
2022-03-02 21:32:33 +00:00
Cluster cluster(name);
2022-02-24 23:02:53 +00:00
if (obj.contains("Status")) {
QString status = obj["Status"].toString();
2022-03-02 21:32:33 +00:00
cluster.setStatus(status);
2022-02-24 23:02:53 +00:00
}
if (!obj.contains("Config")) {
2022-03-02 21:32:33 +00:00
clusters << cluster;
2022-02-24 23:02:53 +00:00
continue;
}
QJsonObject config = obj["Config"].toObject();
if (config.contains("CPUs")) {
int cpus = config["CPUs"].toInt();
2022-03-02 21:32:33 +00:00
cluster.setCpus(cpus);
2022-02-24 23:02:53 +00:00
}
if (config.contains("Memory")) {
int memory = config["Memory"].toInt();
2022-03-02 21:32:33 +00:00
cluster.setMemory(memory);
2022-02-24 23:02:53 +00:00
}
if (config.contains("Driver")) {
QString driver = config["Driver"].toString();
2022-03-02 21:32:33 +00:00
cluster.setDriver(driver);
2022-02-24 23:02:53 +00:00
}
if (!config.contains("KubernetesConfig")) {
2022-03-02 21:32:33 +00:00
clusters << cluster;
2022-02-24 23:02:53 +00:00
continue;
}
QJsonObject k8sConfig = config["KubernetesConfig"].toObject();
if (k8sConfig.contains("ContainerRuntime")) {
QString containerRuntime = k8sConfig["ContainerRuntime"].toString();
2022-03-02 21:32:33 +00:00
cluster.setContainerRuntime(containerRuntime);
2022-02-24 23:02:53 +00:00
}
2022-03-02 21:32:33 +00:00
clusters << cluster;
2022-02-24 23:02:53 +00:00
}
}
}
2022-03-02 21:32:33 +00:00
return clusters;
2022-02-24 23:02:53 +00:00
}
2022-03-02 21:32:33 +00:00
QString Window::selectedCluster()
2022-02-24 23:02:53 +00:00
{
2022-03-02 21:32:33 +00:00
QModelIndex index = clusterListView->currentIndex();
2022-02-24 23:02:53 +00:00
QVariant variant = index.data(Qt::DisplayRole);
if (variant.isNull()) {
return QString();
}
return variant.toString();
}
2022-03-02 21:32:33 +00:00
void Window::setSelectedCluster(QString cluster)
2022-02-24 23:02:53 +00:00
{
2022-03-02 21:32:33 +00:00
QAbstractItemModel *model = clusterListView->model();
2022-02-24 23:02:53 +00:00
QModelIndex start = model->index(0, 0);
2022-03-02 21:32:33 +00:00
QModelIndexList index = model->match(start, Qt::DisplayRole, cluster);
2022-02-24 23:02:53 +00:00
if (index.size() == 0) {
return;
}
2022-03-02 21:32:33 +00:00
clusterListView->setCurrentIndex(index[0]);
2022-02-24 23:02:53 +00:00
}
2022-03-02 21:32:33 +00:00
void Window::createClusterGroupBox()
2022-02-24 23:02:53 +00:00
{
2022-03-02 21:32:33 +00:00
clusterGroupBox = new QGroupBox(tr("Clusters"));
2022-02-24 23:02:53 +00:00
2022-03-02 21:32:33 +00:00
ClusterList clusters = getClusters();
clusterModel = new ClusterModel(clusters);
2022-02-24 23:02:53 +00:00
2022-03-02 21:32:33 +00:00
clusterListView = new QTableView();
clusterListView->setModel(clusterModel);
clusterListView->setSelectionMode(QAbstractItemView::SingleSelection);
clusterListView->setSelectionBehavior(QAbstractItemView::SelectRows);
clusterListView->horizontalHeader()->setSectionResizeMode(0, QHeaderView::Stretch);
clusterListView->horizontalHeader()->setSectionResizeMode(1, QHeaderView::ResizeToContents);
clusterListView->horizontalHeader()->setSectionResizeMode(2, QHeaderView::ResizeToContents);
clusterListView->horizontalHeader()->setSectionResizeMode(3, QHeaderView::ResizeToContents);
clusterListView->horizontalHeader()->setSectionResizeMode(4, QHeaderView::ResizeToContents);
clusterListView->horizontalHeader()->setSectionResizeMode(5, QHeaderView::ResizeToContents);
setSelectedCluster("default");
2022-02-24 23:02:53 +00:00
2022-03-02 21:32:33 +00:00
connect(clusterListView, SIGNAL(clicked(QModelIndex)), this, SLOT(updateButtons()));
2022-02-24 23:02:53 +00:00
startButton = new QPushButton(tr("Start"));
stopButton = new QPushButton(tr("Stop"));
deleteButton = new QPushButton(tr("Delete"));
refreshButton = new QPushButton(tr("Refresh"));
createButton = new QPushButton(tr("Create"));
2022-03-02 23:43:14 +00:00
sshButton = new QPushButton(tr("SSH"));
dashboardButton = new QPushButton(tr("Dashboard"));
2022-02-24 23:02:53 +00:00
updateButtons();
2022-03-02 23:43:14 +00:00
QHBoxLayout *topButtonLayout = new QHBoxLayout;
topButtonLayout->addWidget(createButton);
topButtonLayout->addWidget(refreshButton);
topButtonLayout->addSpacing(340);
QHBoxLayout *bottomButtonLayout = new QHBoxLayout;
bottomButtonLayout->addWidget(startButton);
bottomButtonLayout->addWidget(stopButton);
bottomButtonLayout->addWidget(deleteButton);
bottomButtonLayout->addWidget(sshButton);
bottomButtonLayout->addWidget(dashboardButton);
2022-03-02 21:32:33 +00:00
QVBoxLayout *clusterLayout = new QVBoxLayout;
2022-03-02 23:43:14 +00:00
clusterLayout->addLayout(topButtonLayout);
2022-03-02 21:32:33 +00:00
clusterLayout->addWidget(clusterListView);
2022-03-02 23:43:14 +00:00
clusterLayout->addLayout(bottomButtonLayout);
2022-03-02 21:32:33 +00:00
clusterGroupBox->setLayout(clusterLayout);
2022-02-24 23:02:53 +00:00
}
void Window::updateButtons()
{
2022-03-02 21:32:33 +00:00
QString cluster = selectedCluster();
if (cluster.isEmpty()) {
2022-02-24 23:02:53 +00:00
startButton->setEnabled(false);
stopButton->setEnabled(false);
deleteButton->setEnabled(false);
2022-03-02 23:43:14 +00:00
sshButton->setEnabled(false);
dashboardButton->setEnabled(false);
2022-02-24 23:02:53 +00:00
return;
}
deleteButton->setEnabled(true);
2022-03-02 21:32:33 +00:00
Cluster clusterHash = getClusterHash()[cluster];
if (clusterHash.status() == "Running") {
2022-02-24 23:02:53 +00:00
startButton->setEnabled(false);
stopButton->setEnabled(true);
2022-03-03 01:14:25 +00:00
#if __linux__
2022-03-02 23:43:14 +00:00
sshButton->setEnabled(true);
2022-03-03 01:14:25 +00:00
#endif
2022-03-02 23:43:14 +00:00
dashboardButton->setEnabled(true);
2022-02-24 23:02:53 +00:00
} else {
startButton->setEnabled(true);
stopButton->setEnabled(false);
}
}
2022-03-02 21:32:33 +00:00
ClusterHash Window::getClusterHash()
2022-02-24 23:02:53 +00:00
{
2022-03-02 21:32:33 +00:00
ClusterList clusters = getClusters();
ClusterHash clusterHash;
for (int i = 0; i < clusters.size(); i++) {
Cluster cluster = clusters.at(i);
clusterHash[cluster.name()] = cluster;
2022-02-24 23:02:53 +00:00
}
2022-03-02 21:32:33 +00:00
return clusterHash;
2022-02-24 23:02:53 +00:00
}
bool Window::sendMinikubeCommand(QStringList cmds)
{
QString text;
return sendMinikubeCommand(cmds, text);
}
bool Window::sendMinikubeCommand(QStringList cmds, QString &text)
{
2022-03-02 23:43:14 +00:00
QString program = minikubePath();
if (program.isEmpty()) {
return false;
}
2022-02-24 23:02:53 +00:00
QStringList arguments;
arguments << cmds;
QProcess *process = new QProcess(this);
process->start(program, arguments);
this->setCursor(Qt::WaitCursor);
bool timedOut = process->waitForFinished(300 * 1000);
int exitCode = process->exitCode();
bool success = !timedOut && exitCode == 0;
2022-02-24 23:02:53 +00:00
this->unsetCursor();
text = process->readAllStandardOutput();
2022-02-24 23:02:53 +00:00
if (success) {
} else {
qDebug() << process->readAllStandardError();
}
delete process;
return success;
}
static QString profile = "minikube";
static int cpus = 2;
static int memory = 2400;
static QString driver = "";
static QString containerRuntime = "";
void Window::askName()
{
QDialog dialog;
2022-03-02 21:32:33 +00:00
dialog.setWindowTitle(tr("Create minikube Cluster"));
dialog.setWindowIcon(*trayIconIcon);
2022-02-24 23:02:53 +00:00
dialog.setModal(true);
QFormLayout form(&dialog);
QDialogButtonBox buttonBox(Qt::Horizontal, &dialog);
QLineEdit profileField(profile, &dialog);
form.addRow(new QLabel(tr("Profile")), &profileField);
buttonBox.addButton(QString(tr("Use Default Values")), QDialogButtonBox::AcceptRole);
connect(&buttonBox, &QDialogButtonBox::accepted, &dialog, &QDialog::accept);
buttonBox.addButton(QString(tr("Set Custom Values")), QDialogButtonBox::RejectRole);
connect(&buttonBox, &QDialogButtonBox::rejected, &dialog, &QDialog::reject);
form.addRow(&buttonBox);
int code = dialog.exec();
2022-03-02 23:43:14 +00:00
profile = profileField.text();
2022-02-24 23:02:53 +00:00
if (code == QDialog::Accepted) {
2022-04-06 22:28:18 +00:00
QStringList args = { "-p", profile };
startMinikube(args);
2022-02-24 23:02:53 +00:00
} else if (code == QDialog::Rejected) {
askCustom();
}
}
void Window::askCustom()
{
QDialog dialog;
2022-03-02 21:32:33 +00:00
dialog.setWindowTitle(tr("Set Cluster Values"));
dialog.setWindowIcon(*trayIconIcon);
2022-02-24 23:02:53 +00:00
dialog.setModal(true);
QFormLayout form(&dialog);
driverComboBox = new QComboBox;
driverComboBox->addItem("docker");
2022-03-03 17:50:47 +00:00
#if __linux__
driverComboBox->addItem("kvm2");
#elif __APPLE__
2022-02-24 23:02:53 +00:00
driverComboBox->addItem("hyperkit");
2022-03-03 17:50:47 +00:00
driverComboBox->addItem("parallels");
#else
driverComboBox->addItem("hyperv");
#endif
driverComboBox->addItem("virtualbox");
driverComboBox->addItem("vmware");
driverComboBox->addItem("podman");
2022-02-24 23:02:53 +00:00
form.addRow(new QLabel(tr("Driver")), driverComboBox);
containerRuntimeComboBox = new QComboBox;
containerRuntimeComboBox->addItem("docker");
containerRuntimeComboBox->addItem("containerd");
containerRuntimeComboBox->addItem("crio");
form.addRow(new QLabel(tr("Container Runtime")), containerRuntimeComboBox);
QLineEdit cpuField(QString::number(cpus), &dialog);
form.addRow(new QLabel(tr("CPUs")), &cpuField);
QLineEdit memoryField(QString::number(memory), &dialog);
form.addRow(new QLabel(tr("Memory")), &memoryField);
QDialogButtonBox buttonBox(Qt::Horizontal, &dialog);
buttonBox.addButton(QString(tr("Create")), QDialogButtonBox::AcceptRole);
connect(&buttonBox, &QDialogButtonBox::accepted, &dialog, &QDialog::accept);
buttonBox.addButton(QString(tr("Cancel")), QDialogButtonBox::RejectRole);
connect(&buttonBox, &QDialogButtonBox::rejected, &dialog, &QDialog::reject);
form.addRow(&buttonBox);
int code = dialog.exec();
if (code == QDialog::Accepted) {
driver = driverComboBox->itemText(driverComboBox->currentIndex());
2022-03-02 23:53:42 +00:00
containerRuntime =
containerRuntimeComboBox->itemText(containerRuntimeComboBox->currentIndex());
2022-02-24 23:02:53 +00:00
cpus = cpuField.text().toInt();
memory = memoryField.text().toInt();
2022-04-06 22:28:18 +00:00
QStringList args = { "-p",
2022-03-02 23:53:42 +00:00
profile,
"--driver",
driver,
"--container-runtime",
containerRuntime,
"--cpus",
QString::number(cpus),
"--memory",
2022-04-06 22:32:17 +00:00
QString::number(memory) };
2022-04-06 22:28:18 +00:00
startMinikube(args);
}
}
2022-04-06 22:32:17 +00:00
void Window::outputFailedStart(QString text)
{
QStringList lines;
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
lines = text.split("\n", Qt::SkipEmptyParts);
#else
lines = text.split("\n", QString::SkipEmptyParts);
#endif
for (int i = 0; i < lines.size(); i++) {
QString line = lines.at(i);
QJsonParseError error;
QJsonDocument json = QJsonDocument::fromJson(line.toUtf8(), &error);
if (json.isNull() || !json.isObject()) {
continue;
}
QJsonObject par = json.object();
QJsonObject data = par["data"].toObject();
if (!data.contains("exitcode")) {
continue;
}
QString advice = data["advice"].toString();
QString message = data["message"].toString();
QString name = data["name"].toString();
QString url = data["url"].toString();
QString issues = data["issues"].toString();
QDialog dialog;
dialog.setWindowTitle(tr("minikube start failed"));
dialog.setWindowIcon(*trayIconIcon);
dialog.setFixedWidth(600);
dialog.setModal(true);
QFormLayout form(&dialog);
QLabel *errorCodeLabel = new QLabel(this);
errorCodeLabel->setWordWrap(true);
if (!name.isEmpty()) {
errorCodeLabel->setText("Error Code: " + name);
form.addRow(errorCodeLabel);
}
QLabel *adviceLabel = new QLabel(this);
adviceLabel->setWordWrap(true);
if (!advice.isEmpty()) {
adviceLabel->setText("Advice: " + advice);
form.addRow(adviceLabel);
}
QLabel *messageLabel = new QLabel(this);
messageLabel->setWordWrap(true);
if (!message.isEmpty()) {
messageLabel->setText("Error message: " + message);
form.addRow(messageLabel);
}
QLabel *urlLabel = new QLabel(this);
urlLabel->setOpenExternalLinks(true);
urlLabel->setWordWrap(true);
if (!url.isEmpty()) {
urlLabel->setText("Link to documentation: <a href='" + url + "'>" + url + "</a>");
form.addRow(urlLabel);
}
QLabel *linkLabel = new QLabel(this);
linkLabel->setOpenExternalLinks(true);
urlLabel->setWordWrap(true);
if (!issues.isEmpty()) {
urlLabel->setText("Link to related issue: <a href='" + issues + "'>" + issues + "</a>");
form.addRow(urlLabel);
}
QDialogButtonBox buttonBox(Qt::Horizontal, &dialog);
buttonBox.addButton(QString(tr("OK")), QDialogButtonBox::AcceptRole);
connect(&buttonBox, &QDialogButtonBox::accepted, &dialog, &QDialog::accept);
form.addRow(&buttonBox);
dialog.exec();
2022-02-24 23:02:53 +00:00
}
}
void Window::initMachine()
{
askName();
2022-03-02 21:32:33 +00:00
updateClusters();
2022-02-24 23:02:53 +00:00
}
2022-03-02 23:43:14 +00:00
void Window::sshConsole()
{
QString program = minikubePath();
#ifndef QT_NO_TERMWIDGET
QMainWindow *mainWindow = new QMainWindow();
int startnow = 0; // set shell program first
QTermWidget *console = new QTermWidget(startnow);
QFont font = QApplication::font();
font.setFamily("Monospace");
font.setPointSize(10);
console->setTerminalFont(font);
console->setColorScheme("Tango");
console->setShellProgram(program);
QStringList args = { "ssh" };
console->setArgs(args);
console->startShellProgram();
QObject::connect(console, SIGNAL(finished()), mainWindow, SLOT(close()));
mainWindow->setWindowTitle(nameLabel->text());
mainWindow->resize(800, 400);
mainWindow->setCentralWidget(console);
mainWindow->show();
#else
QString terminal = qEnvironmentVariable("TERMINAL");
if (terminal.isEmpty()) {
terminal = "x-terminal-emulator";
if (QStandardPaths::findExecutable(terminal).isEmpty()) {
terminal = "xterm";
}
}
2022-03-02 23:53:42 +00:00
QStringList arguments = { "-e", QString("%1 ssh -p %2").arg(program, selectedCluster()) };
2022-03-02 23:43:14 +00:00
QProcess *process = new QProcess(this);
process->start(QStandardPaths::findExecutable(terminal), arguments);
#endif
}
void Window::dashboardBrowser()
{
dashboardClose();
QString program = minikubePath();
QProcess *process = new QProcess(this);
2022-03-02 23:53:42 +00:00
QStringList arguments = { "dashboard", "-p", selectedCluster() };
2022-03-02 23:43:14 +00:00
process->start(program, arguments);
dashboardProcess = process;
dashboardProcess->waitForStarted();
}
void Window::dashboardClose()
{
if (dashboardProcess) {
dashboardProcess->terminate();
dashboardProcess->waitForFinished();
}
}
void Window::checkForMinikube()
{
QString program = minikubePath();
if (!program.isEmpty()) {
return;
}
QDialog dialog;
dialog.setWindowTitle(tr("minikube"));
dialog.setWindowIcon(*trayIconIcon);
dialog.setModal(true);
QFormLayout form(&dialog);
QLabel *message = new QLabel(this);
message->setText("minikube was not found on the path.\nPlease follow the install instructions "
"below to install minikube first.\n");
form.addWidget(message);
QLabel *link = new QLabel(this);
link->setOpenExternalLinks(true);
link->setText("<a "
"href='https://minikube.sigs.k8s.io/docs/start/'>https://minikube.sigs.k8s.io/"
"docs/start/</a>");
form.addWidget(link);
QDialogButtonBox buttonBox(Qt::Horizontal, &dialog);
buttonBox.addButton(QString(tr("OK")), QDialogButtonBox::AcceptRole);
connect(&buttonBox, &QDialogButtonBox::accepted, &dialog, &QDialog::accept);
form.addRow(&buttonBox);
dialog.exec();
exit(EXIT_FAILURE);
}
2022-02-24 23:02:53 +00:00
#endif