/* * Copyright (C) 2019 KeePassXC Team * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 2 or (at your option) * version 3 of the License. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "Clip.h" #include "Utils.h" #include "core/EntrySearcher.h" #include "core/Group.h" #include "core/Tools.h" #include #define CLI_DEFAULT_CLIP_TIMEOUT 10 const QCommandLineOption Clip::AttributeOption = QCommandLineOption( QStringList() << "a" << "attribute", QObject::tr("Copy the given attribute to the clipboard. Defaults to \"password\" if not specified.", "Don't translate \"password\", it refers to the attribute."), "attr", "password"); const QCommandLineOption Clip::TotpOption = QCommandLineOption(QStringList() << "t" << "totp", QObject::tr("Copy the current TOTP to the clipboard (equivalent to \"-a totp\").")); const QCommandLineOption Clip::BestMatchOption = QCommandLineOption(QStringList() << "b" << "best-match", QObject::tr("Must match only one entry, otherwise a list of possible matches is shown.")); Clip::Clip() { name = QString("clip"); description = QObject::tr("Copy an entry's attribute to the clipboard."); options.append(Clip::AttributeOption); options.append(Clip::TotpOption); options.append(Clip::BestMatchOption); positionalArguments.append( {QString("entry"), QObject::tr("Path of the entry to clip.", "clip = copy to clipboard"), QString("")}); optionalArguments.append( {QString("timeout"), QObject::tr("Timeout before clearing the clipboard (default is %1 seconds, set to 0 for unlimited).") .arg(CLI_DEFAULT_CLIP_TIMEOUT), QString("[timeout]")}); } int Clip::executeWithDatabase(QSharedPointer database, QSharedPointer parser) { auto& out = parser->isSet(Command::QuietOption) ? Utils::DEVNULL : Utils::STDOUT; auto& err = Utils::STDERR; const QStringList args = parser->positionalArguments(); auto timeout = CLI_DEFAULT_CLIP_TIMEOUT; if (args.size() == 3) { bool ok; timeout = args.at(2).toInt(&ok); if (!ok) { err << QObject::tr("Invalid timeout value %1.").arg(args.at(2)) << endl; return EXIT_FAILURE; } } QString entryPath; if (parser->isSet(Clip::BestMatchOption)) { EntrySearcher searcher; const auto& searchTerm = args.at(1); const auto results = searcher.search(QString("title:%1").arg(searchTerm), database->rootGroup(), true); if (results.count() > 1) { err << QObject::tr("Multiple entries matching:") << endl; for (const Entry* result : results) { err << result->path().prepend('/') << endl; } return EXIT_FAILURE; } else { entryPath = (results.isEmpty()) ? searchTerm : results[0]->path().prepend('/'); out << QObject::tr("Using matching entry: %1").arg(entryPath) << endl; } } else { entryPath = args.at(1); } auto* entry = database->rootGroup()->findEntryByPath(entryPath); if (!entry) { err << QObject::tr("Entry %1 not found.").arg(entryPath) << endl; return EXIT_FAILURE; } if (parser->isSet(AttributeOption) && parser->isSet(TotpOption)) { err << QObject::tr("ERROR: Please specify one of --attribute or --totp, not both.") << endl; return EXIT_FAILURE; } QString selectedAttribute = parser->value(AttributeOption); QString value; bool found = false; if (parser->isSet(TotpOption) || selectedAttribute == "totp") { if (!entry->hasTotp()) { err << QObject::tr("Entry with path %1 has no TOTP set up.").arg(entryPath) << endl; return EXIT_FAILURE; } selectedAttribute = "totp"; found = true; value = entry->totp(); } else if (Utils::EntryFieldNames.contains(selectedAttribute)) { value = Utils::getTopLevelField(entry, selectedAttribute); found = true; } else { QStringList attrs = Utils::findAttributes(*entry->attributes(), selectedAttribute); if (attrs.size() > 1) { err << QObject::tr("ERROR: attribute %1 is ambiguous, it matches %2.") .arg(selectedAttribute, QLocale().createSeparatedList(attrs)) << endl; return EXIT_FAILURE; } else if (attrs.size() == 1) { found = true; selectedAttribute = attrs[0]; value = entry->attributes()->value(selectedAttribute); } } if (!found) { out << QObject::tr("Attribute \"%1\" not found.").arg(selectedAttribute) << endl; return EXIT_FAILURE; } int exitCode = Utils::clipText(value); if (exitCode != EXIT_SUCCESS) { return exitCode; } out << QObject::tr("Entry's \"%1\" attribute copied to the clipboard!").arg(selectedAttribute) << endl; if (timeout <= 0) { return exitCode; } QString lastLine = ""; while (timeout > 0) { out << '\r' << QString(lastLine.size(), ' ') << '\r'; lastLine = QObject::tr("Clearing the clipboard in %1 second(s)...", "", timeout).arg(timeout); out << lastLine << flush; Tools::sleep(1000); --timeout; } Utils::clipText(""); out << '\r' << QString(lastLine.size(), ' ') << '\r'; out << QObject::tr("Clipboard cleared!") << endl; return EXIT_SUCCESS; }