parent
73d3ada062
commit
66ccff86ad
@ -1,122 +0,0 @@
|
|||||||
Description: Enable bidirectional communication
|
|
||||||
lxqt-sudo shows you what an executed process prints to stdout/stderr, but it
|
|
||||||
doesn't send anything written to lxqt-sudo's stdin to the child process. This
|
|
||||||
patch fixes that and adds some small miscellaneous improvements.
|
|
||||||
Origin: https://github.com/ArrayBolt3/lxqt-sudo/compare/d69855c2900d10c2aae4d39678b71aadf37dc37c...4f430cb0fe240566e74ade15e1657b1f99177c12
|
|
||||||
Forwarded: https://github.com/lxqt/lxqt-sudo/pull/210
|
|
||||||
Last-Update: 2024-01-11
|
|
||||||
---
|
|
||||||
This patch header follows DEP-3: http://dep.debian.net/deps/dep3/
|
|
||||||
diff --git a/sudo.cpp b/sudo.cpp
|
|
||||||
index d12f77c..eebcf5c 100644
|
|
||||||
--- a/sudo.cpp
|
|
||||||
+++ b/sudo.cpp
|
|
||||||
@@ -68,6 +68,7 @@ namespace
|
|
||||||
const QString doas_prog{QStringLiteral(LXQTSUDO_DOAS)};
|
|
||||||
const QString pwd_prompt_end{QStringLiteral(": ")};
|
|
||||||
const QChar nl{QLatin1Char('\n')};
|
|
||||||
+ constexpr int term_eol_size = 2;
|
|
||||||
|
|
||||||
void usage(QString const & err = QString())
|
|
||||||
{
|
|
||||||
@@ -374,12 +375,14 @@ int Sudo::parent()
|
|
||||||
}
|
|
||||||
|
|
||||||
QTextStream child_str{pwd_f};
|
|
||||||
+ // pseudoterminal echoes everything written into it's input; we don't want duplicating input
|
|
||||||
+ int inhibit_count = 0;
|
|
||||||
|
|
||||||
QObject::connect(mDlg.data(), &QDialog::finished, [&] (int result)
|
|
||||||
{
|
|
||||||
if (QDialog::Accepted == result)
|
|
||||||
{
|
|
||||||
- child_str << mDlg->password().append(nl);
|
|
||||||
+ child_str << mDlg->password() << nl;
|
|
||||||
child_str.flush();
|
|
||||||
} else
|
|
||||||
{
|
|
||||||
@@ -388,6 +391,8 @@ int Sudo::parent()
|
|
||||||
});
|
|
||||||
|
|
||||||
QString last_line;
|
|
||||||
+ QString const & error_check = QStringLiteral("%1:").arg(backendName());
|
|
||||||
+ QTextStream stderr_str{stderr, QIODevice::WriteOnly};
|
|
||||||
QScopedPointer<QSocketNotifier> pwd_watcher{new QSocketNotifier{mPwdFd, QSocketNotifier::Read}};
|
|
||||||
auto reader = [&]
|
|
||||||
{
|
|
||||||
@@ -396,11 +401,10 @@ int Sudo::parent()
|
|
||||||
{
|
|
||||||
pwd_watcher.reset(nullptr); //stop the notifications events
|
|
||||||
|
|
||||||
- QString const & prog = backendName();
|
|
||||||
- if (last_line.startsWith(QStringLiteral("%1:").arg(prog)))
|
|
||||||
+ if (last_line.startsWith(error_check))
|
|
||||||
{
|
|
||||||
QMessageBox(QMessageBox::Critical, mDlg->windowTitle()
|
|
||||||
- , tr("Child '%1' process failed!\n%2").arg(prog).arg(last_line), QMessageBox::Ok).exec();
|
|
||||||
+ , tr("Child '%1' process failed!\n%2").arg(backendName()).arg(last_line), QMessageBox::Ok).exec();
|
|
||||||
}
|
|
||||||
} else
|
|
||||||
{
|
|
||||||
@@ -414,18 +418,56 @@ int Sudo::parent()
|
|
||||||
if (!(ECHO & tios.c_lflag))
|
|
||||||
{
|
|
||||||
mDlg->show();
|
|
||||||
- return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
- QTextStream{stderr, QIODevice::WriteOnly} << line;
|
|
||||||
- //assuming text oriented output
|
|
||||||
- QStringList lines = line.split(nl, Qt::SkipEmptyParts);
|
|
||||||
- last_line = lines.isEmpty() ? QString() : lines.back();
|
|
||||||
+ if (inhibit_count > 0)
|
|
||||||
+ {
|
|
||||||
+ if (inhibit_count < line.count())
|
|
||||||
+ {
|
|
||||||
+ stderr_str << line.right(line.count() - inhibit_count);
|
|
||||||
+ stderr_str.flush();
|
|
||||||
+ inhibit_count = 0;
|
|
||||||
+ } else
|
|
||||||
+ {
|
|
||||||
+ inhibit_count -= line.count();
|
|
||||||
+ }
|
|
||||||
+ } else
|
|
||||||
+ {
|
|
||||||
+ stderr_str << line;
|
|
||||||
+ stderr_str.flush();
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ //assuming text oriented output; find the last non-empty line
|
|
||||||
+ auto i = line.crbegin(), i_end = line.crbegin(), i_crend = line.crend();
|
|
||||||
+ do {
|
|
||||||
+ i_end = i + 1;
|
|
||||||
+ i = std::find(i_end, i_crend, nl);
|
|
||||||
+ } while (i != i_crend && std::distance(i, i_end) == 0);
|
|
||||||
+
|
|
||||||
+ last_line.clear();
|
|
||||||
+ last_line.reserve(std::distance(i, i_end));
|
|
||||||
+ std::for_each(i.base(), i_end.base(), [&last_line](decltype (*i.base()) val) { last_line.append(val); });
|
|
||||||
}
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
+ QTextStream stdin_str{stdin, QIODevice::ReadOnly};
|
|
||||||
+ QScopedPointer<QSocketNotifier> stdin_watcher{new QSocketNotifier{STDIN_FILENO, QSocketNotifier::Read}};
|
|
||||||
+ auto writer = [&]
|
|
||||||
+ {
|
|
||||||
+ QString line = stdin_str.readLine();
|
|
||||||
+ if (line.isEmpty()) {
|
|
||||||
+ stdin_watcher.reset(nullptr); //stop the notification events
|
|
||||||
+ } else
|
|
||||||
+ {
|
|
||||||
+ inhibit_count += line.count() + term_eol_size;
|
|
||||||
+ child_str << line << nl;
|
|
||||||
+ child_str.flush();
|
|
||||||
+ }
|
|
||||||
+ };
|
|
||||||
+
|
|
||||||
QObject::connect(pwd_watcher.data(), &QSocketNotifier::activated, reader);
|
|
||||||
+ QObject::connect(stdin_watcher.data(), &QSocketNotifier::activated, writer);
|
|
||||||
|
|
||||||
std::unique_ptr<std::thread> child_waiter;
|
|
||||||
QTimer::singleShot(0, [&child_waiter, this] {
|
|
Loading…
Reference in new issue