mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-04-05 13:37:43 +03:00
Feature : Clip command using native programs. (#792)
* Adding a timeout option * Using native apps. * Renaming PasswordInput -> Utils
This commit is contained in:
parent
90468e8095
commit
1edabc4b3c
7 changed files with 105 additions and 28 deletions
|
@ -66,8 +66,8 @@ set(keepassx_SOURCES
|
||||||
core/Tools.cpp
|
core/Tools.cpp
|
||||||
core/Translator.cpp
|
core/Translator.cpp
|
||||||
core/Uuid.cpp
|
core/Uuid.cpp
|
||||||
cli/PasswordInput.cpp
|
cli/Utils.cpp
|
||||||
cli/PasswordInput.h
|
cli/Utils.h
|
||||||
crypto/Crypto.cpp
|
crypto/Crypto.cpp
|
||||||
crypto/CryptoHash.cpp
|
crypto/CryptoHash.cpp
|
||||||
crypto/Random.cpp
|
crypto/Random.cpp
|
||||||
|
|
|
@ -15,8 +15,10 @@
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <chrono>
|
||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
#include <thread>
|
||||||
|
|
||||||
#include "Clip.h"
|
#include "Clip.h"
|
||||||
|
|
||||||
|
@ -25,11 +27,11 @@
|
||||||
#include <QStringList>
|
#include <QStringList>
|
||||||
#include <QTextStream>
|
#include <QTextStream>
|
||||||
|
|
||||||
#include "gui/UnlockDatabaseDialog.h"
|
#include "cli/Utils.h"
|
||||||
#include "core/Database.h"
|
#include "core/Database.h"
|
||||||
#include "core/Entry.h"
|
#include "core/Entry.h"
|
||||||
#include "core/Group.h"
|
#include "core/Group.h"
|
||||||
#include "gui/Clipboard.h"
|
#include "gui/UnlockDatabaseDialog.h"
|
||||||
|
|
||||||
Clip::Clip()
|
Clip::Clip()
|
||||||
{
|
{
|
||||||
|
@ -50,6 +52,7 @@ int Clip::execute(int argc, char** argv)
|
||||||
}
|
}
|
||||||
|
|
||||||
QTextStream out(stdout);
|
QTextStream out(stdout);
|
||||||
|
QApplication app(argc, argv);
|
||||||
|
|
||||||
QCommandLineParser parser;
|
QCommandLineParser parser;
|
||||||
parser.setApplicationDescription(this->description);
|
parser.setApplicationDescription(this->description);
|
||||||
|
@ -59,17 +62,19 @@ int Clip::execute(int argc, char** argv)
|
||||||
QObject::tr("Use a GUI prompt unlocking the database."));
|
QObject::tr("Use a GUI prompt unlocking the database."));
|
||||||
parser.addOption(guiPrompt);
|
parser.addOption(guiPrompt);
|
||||||
parser.addPositionalArgument("entry", QObject::tr("Path of the entry to clip."));
|
parser.addPositionalArgument("entry", QObject::tr("Path of the entry to clip."));
|
||||||
|
parser.addPositionalArgument(
|
||||||
|
"timeout",
|
||||||
|
QObject::tr("Timeout in seconds before clearing the clipboard."),
|
||||||
|
QString("[timeout]"));
|
||||||
parser.process(arguments);
|
parser.process(arguments);
|
||||||
|
|
||||||
const QStringList args = parser.positionalArguments();
|
const QStringList args = parser.positionalArguments();
|
||||||
if (args.size() != 2) {
|
if (args.size() != 2 && args.size() != 3) {
|
||||||
QCoreApplication app(argc, argv);
|
|
||||||
out << parser.helpText().replace("keepassxc-cli", "keepassxc-cli clip");
|
out << parser.helpText().replace("keepassxc-cli", "keepassxc-cli clip");
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
|
|
||||||
Database* db = nullptr;
|
Database* db = nullptr;
|
||||||
QApplication app(argc, argv);
|
|
||||||
if (parser.isSet("gui-prompt")) {
|
if (parser.isSet("gui-prompt")) {
|
||||||
db = UnlockDatabaseDialog::openDatabasePrompt(args.at(0));
|
db = UnlockDatabaseDialog::openDatabasePrompt(args.at(0));
|
||||||
} else {
|
} else {
|
||||||
|
@ -79,12 +84,20 @@ int Clip::execute(int argc, char** argv)
|
||||||
if (!db) {
|
if (!db) {
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
return this->clipEntry(db, args.at(1));
|
return this->clipEntry(db, args.at(1), args.value(2));
|
||||||
}
|
}
|
||||||
|
|
||||||
int Clip::clipEntry(Database* database, QString entryPath)
|
int Clip::clipEntry(Database* database, QString entryPath, QString timeout)
|
||||||
{
|
{
|
||||||
|
|
||||||
|
int timeoutSeconds = 0;
|
||||||
|
if (!timeout.isEmpty() && !timeout.toInt()) {
|
||||||
|
qCritical("Invalid timeout value %s.", qPrintable(timeout));
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
} else if (!timeout.isEmpty()) {
|
||||||
|
timeoutSeconds = timeout.toInt();
|
||||||
|
}
|
||||||
|
|
||||||
QTextStream outputTextStream(stdout, QIODevice::WriteOnly);
|
QTextStream outputTextStream(stdout, QIODevice::WriteOnly);
|
||||||
Entry* entry = database->rootGroup()->findEntry(entryPath);
|
Entry* entry = database->rootGroup()->findEntry(entryPath);
|
||||||
if (!entry) {
|
if (!entry) {
|
||||||
|
@ -92,7 +105,25 @@ int Clip::clipEntry(Database* database, QString entryPath)
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
|
|
||||||
Clipboard::instance()->setText(entry->password());
|
int exitCode = Utils::clipText(entry->password());
|
||||||
return EXIT_SUCCESS;
|
if (exitCode != EXIT_SUCCESS) {
|
||||||
|
return exitCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
outputTextStream << "Entry's password copied to the clipboard!" << endl;
|
||||||
|
|
||||||
|
if (!timeoutSeconds) {
|
||||||
|
return exitCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (timeoutSeconds > 0) {
|
||||||
|
outputTextStream << "\rClearing the clipboard in " << timeoutSeconds << " seconds...";
|
||||||
|
outputTextStream.flush();
|
||||||
|
std::this_thread::sleep_for(std::chrono::milliseconds(1000));
|
||||||
|
timeoutSeconds--;
|
||||||
|
}
|
||||||
|
Utils::clipText("");
|
||||||
|
outputTextStream << "\nClipboard cleared!" << endl;
|
||||||
|
|
||||||
|
return EXIT_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,7 +26,7 @@ public:
|
||||||
Clip();
|
Clip();
|
||||||
~Clip();
|
~Clip();
|
||||||
int execute(int argc, char** argv);
|
int execute(int argc, char** argv);
|
||||||
int clipEntry(Database* database, QString entryPath);
|
int clipEntry(Database* database, QString entryPath, QString timeout);
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // KEEPASSXC_CLIP_H
|
#endif // KEEPASSXC_CLIP_H
|
||||||
|
|
|
@ -26,7 +26,7 @@
|
||||||
#include <QStringList>
|
#include <QStringList>
|
||||||
#include <QTextStream>
|
#include <QTextStream>
|
||||||
|
|
||||||
#include "cli/PasswordInput.h"
|
#include "cli/Utils.h"
|
||||||
#include "core/Database.h"
|
#include "core/Database.h"
|
||||||
#include "format/KeePass2Reader.h"
|
#include "format/KeePass2Reader.h"
|
||||||
#include "keys/CompositeKey.h"
|
#include "keys/CompositeKey.h"
|
||||||
|
@ -66,7 +66,7 @@ int Extract::execute(int argc, char** argv)
|
||||||
out << "Insert the database password\n> ";
|
out << "Insert the database password\n> ";
|
||||||
out.flush();
|
out.flush();
|
||||||
|
|
||||||
QString line = PasswordInput::getPassword();
|
QString line = Utils::getPassword();
|
||||||
CompositeKey key = CompositeKey::readFromLine(line);
|
CompositeKey key = CompositeKey::readFromLine(line);
|
||||||
|
|
||||||
QString databaseFilename = args.at(0);
|
QString databaseFilename = args.at(0);
|
||||||
|
|
|
@ -15,7 +15,7 @@
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "PasswordInput.h"
|
#include "Utils.h"
|
||||||
|
|
||||||
#ifdef Q_OS_WIN
|
#ifdef Q_OS_WIN
|
||||||
#include <windows.h>
|
#include <windows.h>
|
||||||
|
@ -24,14 +24,11 @@
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include <QProcess>
|
||||||
#include <QTextStream>
|
#include <QTextStream>
|
||||||
|
|
||||||
|
|
||||||
PasswordInput::PasswordInput()
|
void Utils::setStdinEcho(bool enable = true)
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
void PasswordInput::setStdinEcho(bool enable = true)
|
|
||||||
{
|
{
|
||||||
#ifdef Q_OS_WIN
|
#ifdef Q_OS_WIN
|
||||||
HANDLE hIn = GetStdHandle(STD_INPUT_HANDLE);
|
HANDLE hIn = GetStdHandle(STD_INPUT_HANDLE);
|
||||||
|
@ -60,7 +57,7 @@ void PasswordInput::setStdinEcho(bool enable = true)
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
QString PasswordInput::getPassword()
|
QString Utils::getPassword()
|
||||||
{
|
{
|
||||||
static QTextStream inputTextStream(stdin, QIODevice::ReadOnly);
|
static QTextStream inputTextStream(stdin, QIODevice::ReadOnly);
|
||||||
static QTextStream outputTextStream(stdout, QIODevice::WriteOnly);
|
static QTextStream outputTextStream(stdout, QIODevice::WriteOnly);
|
||||||
|
@ -75,3 +72,52 @@ QString PasswordInput::getPassword()
|
||||||
|
|
||||||
return line;
|
return line;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* A valid and running event loop is needed to use the global QClipboard,
|
||||||
|
* so we need to use this from the CLI.
|
||||||
|
*/
|
||||||
|
int Utils::clipText(QString text)
|
||||||
|
{
|
||||||
|
|
||||||
|
QString programName = "";
|
||||||
|
QStringList arguments;
|
||||||
|
|
||||||
|
#ifdef Q_OS_UNIX
|
||||||
|
programName = "xclip";
|
||||||
|
arguments << "-i"
|
||||||
|
<< "-selection"
|
||||||
|
<< "clipboard";
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef Q_OS_MACOS
|
||||||
|
programName = "pbcopy";
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef Q_OS_WIN
|
||||||
|
programName = "clip";
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (programName.isEmpty()) {
|
||||||
|
qCritical("No program defined for clipboard manipulation");
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
QProcess* clipProcess = new QProcess(nullptr);
|
||||||
|
clipProcess->start(programName, arguments);
|
||||||
|
clipProcess->waitForStarted();
|
||||||
|
|
||||||
|
if (clipProcess->state() != QProcess::Running) {
|
||||||
|
qCritical("Unable to start program %s", qPrintable(programName));
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (clipProcess->write(text.toLatin1()) == -1) {
|
||||||
|
qDebug("Unable to write to process : %s", qPrintable(clipProcess->errorString()));
|
||||||
|
}
|
||||||
|
clipProcess->waitForBytesWritten();
|
||||||
|
clipProcess->closeWriteChannel();
|
||||||
|
clipProcess->waitForFinished();
|
||||||
|
|
||||||
|
return clipProcess->exitCode();
|
||||||
|
}
|
|
@ -15,17 +15,17 @@
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef KEEPASSXC_PASSWORDINPUT_H
|
#ifndef KEEPASSXC_UTILS_H
|
||||||
#define KEEPASSXC_PASSWORDINPUT_H
|
#define KEEPASSXC_UTILS_H
|
||||||
|
|
||||||
#include <QtCore/qglobal.h>
|
#include <QtCore/qglobal.h>
|
||||||
|
|
||||||
class PasswordInput
|
class Utils
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
PasswordInput();
|
|
||||||
static void setStdinEcho(bool enable);
|
static void setStdinEcho(bool enable);
|
||||||
static QString getPassword();
|
static QString getPassword();
|
||||||
|
static int clipText(QString text);
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // KEEPASSXC_PASSWORDINPUT_H
|
#endif // KEEPASSXC_UTILS_H
|
|
@ -24,7 +24,7 @@
|
||||||
#include <QTimer>
|
#include <QTimer>
|
||||||
#include <QXmlStreamReader>
|
#include <QXmlStreamReader>
|
||||||
|
|
||||||
#include "cli/PasswordInput.h"
|
#include "cli/Utils.h"
|
||||||
#include "core/Group.h"
|
#include "core/Group.h"
|
||||||
#include "core/Metadata.h"
|
#include "core/Metadata.h"
|
||||||
#include "crypto/Random.h"
|
#include "crypto/Random.h"
|
||||||
|
@ -404,7 +404,7 @@ Database* Database::unlockFromStdin(QString databaseFilename)
|
||||||
outputTextStream << QString("Insert password to unlock " + databaseFilename + "\n> ");
|
outputTextStream << QString("Insert password to unlock " + databaseFilename + "\n> ");
|
||||||
outputTextStream.flush();
|
outputTextStream.flush();
|
||||||
|
|
||||||
QString line = PasswordInput::getPassword();
|
QString line = Utils::getPassword();
|
||||||
CompositeKey key = CompositeKey::readFromLine(line);
|
CompositeKey key = CompositeKey::readFromLine(line);
|
||||||
return Database::openDatabaseFile(databaseFilename, key);
|
return Database::openDatabaseFile(databaseFilename, key);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue