File indexing completed on 2024-09-08 13:30:53
0001 // SPDX-FileCopyrightText: 2021 Simon Persson <simon.persson@mykolab.com> 0002 // 0003 // SPDX-License-Identifier: GPL-3.0-only OR LicenseRef-KDE-Accepted-GPL 0004 0005 #include "purger.h" 0006 #include "kuppurger_debug.h" 0007 0008 #include <KFormat> 0009 #include <KLocalizedString> 0010 #include <KStandardAction> 0011 #include <KToolBar> 0012 0013 #include <QDateTime> 0014 #include <QGuiApplication> 0015 #include <QHash> 0016 #include <QSplitter> 0017 0018 Purger::Purger(QString pRepoPath, QString pBranchName, QWidget *pParent) 0019 : KMainWindow(pParent), mRepoPath(std::move(pRepoPath)), 0020 mBranchName(std::move(pBranchName)) 0021 { 0022 setWindowIcon(QIcon::fromTheme(QStringLiteral("kup"))); 0023 KToolBar *lAppToolBar = toolBar(); 0024 lAppToolBar->addAction(KStandardAction::quit(this, SLOT(close()), this)); 0025 mDeleteAction = KStandardAction::deleteFile(this, SLOT(purge()), this); 0026 lAppToolBar->addAction(mDeleteAction); 0027 0028 mListWidget = new QListWidget(); 0029 auto lSplitter = new QSplitter(); 0030 lSplitter->addWidget(mListWidget); 0031 mTextEdit = new QTextEdit(); 0032 mTextEdit->setReadOnly(true); 0033 lSplitter->addWidget(mTextEdit); 0034 setCentralWidget(lSplitter); 0035 0036 mCollectProcess = new KProcess(); 0037 mCollectProcess->setOutputChannelMode(KProcess::SeparateChannels); 0038 *mCollectProcess << QStringLiteral("bup"); 0039 *mCollectProcess << QStringLiteral("-d") << mRepoPath; 0040 *mCollectProcess << QStringLiteral("gc") << QStringLiteral("--unsafe") << QStringLiteral("--verbose"); 0041 connect(mCollectProcess, &KProcess::readyReadStandardError, this, [this] { 0042 auto lLogText = QString::fromUtf8(mCollectProcess->readAllStandardError()); 0043 qCInfo(KUPPURGER) << lLogText; 0044 mTextEdit->append(lLogText); 0045 }); 0046 connect(mCollectProcess, SIGNAL(finished(int,QProcess::ExitStatus)), 0047 SLOT(purgeDone(int,QProcess::ExitStatus))); 0048 0049 mListProcess = new KProcess(); 0050 mListProcess->setOutputChannelMode(KProcess::SeparateChannels); 0051 *mListProcess << QStringLiteral("bup"); 0052 *mListProcess << QStringLiteral("-d") << mRepoPath; 0053 *mListProcess << QStringLiteral("ls") << QStringLiteral("--hash") << mBranchName; 0054 connect(mListProcess, &KProcess::readyReadStandardOutput, this, [this] { 0055 KFormat lFormat; 0056 const auto lLines = QString::fromUtf8(mListProcess->readAllStandardOutput()).split(QChar::LineFeed); 0057 for(const QString &lLine: lLines) { 0058 qCDebug(KUPPURGER) << lLine; 0059 const auto lHash = lLine.left(40); 0060 if(!lHash.isEmpty() && lHash != QStringLiteral("0000000000000000000000000000000000000000")) { 0061 const auto lTimeStamp = lLine.mid(41); 0062 if(mHashes.contains(lHash)) { 0063 auto lItem = mHashes.value(lHash); 0064 lItem->setWhatsThis(lItem->whatsThis() + QChar::LineFeed + lTimeStamp); 0065 } else { 0066 const auto lDateTime = QDateTime::fromString(lTimeStamp, QStringLiteral("yyyy-MM-dd-HHmmss")); 0067 const auto lDisplayText = lFormat.formatRelativeDateTime(lDateTime, QLocale::ShortFormat); 0068 auto lItem = new QListWidgetItem(lDisplayText, mListWidget); 0069 lItem->setWhatsThis(lTimeStamp); //misuse of field, for later use when removing 0070 lItem->setFlags(Qt::ItemIsUserCheckable | Qt::ItemIsEnabled); 0071 lItem->setCheckState(Qt::Unchecked); 0072 mHashes.insert(lHash, lItem); 0073 } 0074 } 0075 } 0076 }); 0077 connect(mListProcess, SIGNAL(finished(int,QProcess::ExitStatus)), 0078 SLOT(listDone(int,QProcess::ExitStatus))); 0079 0080 fillListWidget(); 0081 } 0082 0083 QSize Purger::sizeHint() const { 0084 return {800, 600}; 0085 } 0086 0087 void Purger::fillListWidget() { 0088 QGuiApplication::setOverrideCursor(QCursor(Qt::WaitCursor)); 0089 mListWidget->clear(); 0090 mHashes.clear(); 0091 mDeleteAction->setEnabled(false); 0092 mListProcess->start(); 0093 } 0094 0095 void Purger::listDone(int, QProcess::ExitStatus) { 0096 QGuiApplication::restoreOverrideCursor(); 0097 mDeleteAction->setEnabled(true); 0098 } 0099 0100 void Purger::purge() { 0101 qCInfo(KUPPURGER) << "Starting purge operation."; 0102 mDeleteAction->setEnabled(false); 0103 bool lAnythingRemoved = false; 0104 for(int i=0; i < mListWidget->count(); ++i) { 0105 auto lItem = mListWidget->item(i); 0106 qCInfo(KUPPURGER) << lItem->text() << lItem->whatsThis() << lItem->checkState(); 0107 if(lItem->checkState() == Qt::Checked) { 0108 const auto lTimeStamps = lItem->whatsThis().split(QChar::LineFeed); 0109 for(const QString &lTimeStamp: lTimeStamps) { 0110 KProcess lRemoveProcess; 0111 lRemoveProcess.setOutputChannelMode(KProcess::SeparateChannels); 0112 lRemoveProcess << QStringLiteral("bup"); 0113 lRemoveProcess << QStringLiteral("-d") << mRepoPath << QStringLiteral("rm"); 0114 lRemoveProcess << QStringLiteral("--unsafe") << QStringLiteral("--verbose"); 0115 lRemoveProcess << QString("%1/%2").arg(mBranchName, lTimeStamp); 0116 qCInfo(KUPPURGER) << lRemoveProcess.program(); 0117 if(lRemoveProcess.execute() == 0) { 0118 lAnythingRemoved = true; 0119 qCInfo(KUPPURGER) << "Successfully removed snapshot"; 0120 } 0121 const auto lLogText = QString::fromUtf8(lRemoveProcess.readAllStandardError()); 0122 qCInfo(KUPPURGER) << lLogText; 0123 mTextEdit->append(lLogText); 0124 } 0125 } 0126 } 0127 if(lAnythingRemoved) { 0128 QGuiApplication::setOverrideCursor(QCursor(Qt::WaitCursor)); 0129 qCInfo(KUPPURGER) << mCollectProcess->program(); 0130 mCollectProcess->start(); 0131 } else { 0132 mDeleteAction->setEnabled(true); 0133 } 0134 } 0135 0136 void Purger::purgeDone(int pExitCode, QProcess::ExitStatus pExitStatus) { 0137 qCInfo(KUPPURGER) << pExitCode << pExitStatus; 0138 QGuiApplication::restoreOverrideCursor(); 0139 mDeleteAction->setEnabled(true); 0140 const auto lLogText = QString::fromUtf8(mCollectProcess->readAllStandardError()); 0141 qCInfo(KUPPURGER) << lLogText; 0142 mTextEdit->append(lLogText); 0143 fillListWidget(); 0144 } 0145