mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-04-04 21:17:43 +03:00
parent
9e29b5c7b6
commit
edab0faa94
16 changed files with 587 additions and 38 deletions
|
@ -90,6 +90,7 @@ set(core_SOURCES
|
|||
format/OpVaultReaderAttachments.cpp
|
||||
format/OpVaultReaderBandEntry.cpp
|
||||
format/OpVaultReaderSections.cpp
|
||||
format/ProtonPassReader.cpp
|
||||
keys/CompositeKey.cpp
|
||||
keys/FileKey.cpp
|
||||
keys/PasswordKey.cpp
|
||||
|
|
221
src/format/ProtonPassReader.cpp
Normal file
221
src/format/ProtonPassReader.cpp
Normal file
|
@ -0,0 +1,221 @@
|
|||
/*
|
||||
* Copyright (C) 2024 KeePassXC Team <team@keepassxc.org>
|
||||
*
|
||||
* 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 <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include "ProtonPassReader.h"
|
||||
|
||||
#include "core/Database.h"
|
||||
#include "core/Entry.h"
|
||||
#include "core/Group.h"
|
||||
#include "core/Metadata.h"
|
||||
#include "core/Tools.h"
|
||||
#include "core/Totp.h"
|
||||
#include "crypto/CryptoHash.h"
|
||||
|
||||
#include <QFileInfo>
|
||||
#include <QJsonArray>
|
||||
#include <QJsonDocument>
|
||||
#include <QJsonObject>
|
||||
#include <QJsonParseError>
|
||||
#include <QMap>
|
||||
#include <QScopedPointer>
|
||||
#include <QUrl>
|
||||
|
||||
namespace
|
||||
{
|
||||
Entry* readItem(const QJsonObject& item)
|
||||
{
|
||||
const auto itemMap = item.toVariantMap();
|
||||
const auto dataMap = itemMap.value("data").toMap();
|
||||
const auto metadataMap = dataMap.value("metadata").toMap();
|
||||
|
||||
// Create entry and assign basic values
|
||||
QScopedPointer<Entry> entry(new Entry());
|
||||
entry->setUuid(QUuid::createUuid());
|
||||
entry->setTitle(metadataMap.value("name").toString());
|
||||
entry->setNotes(metadataMap.value("note").toString());
|
||||
|
||||
if (itemMap.value("pinned").toBool()) {
|
||||
entry->addTag(QObject::tr("Favorite", "Tag for favorite entries"));
|
||||
}
|
||||
|
||||
// Handle specific item types
|
||||
auto type = dataMap.value("type").toString();
|
||||
|
||||
// Login
|
||||
if (type.compare("login", Qt::CaseInsensitive) == 0) {
|
||||
const auto loginMap = dataMap.value("content").toMap();
|
||||
entry->setUsername(loginMap.value("itemUsername").toString());
|
||||
entry->setPassword(loginMap.value("password").toString());
|
||||
if (loginMap.contains("totpUri")) {
|
||||
auto totp = loginMap.value("totpUri").toString();
|
||||
if (!totp.startsWith("otpauth://")) {
|
||||
QUrl url(QString("otpauth://totp/%1:%2?secret=%3")
|
||||
.arg(QString(QUrl::toPercentEncoding(entry->title())),
|
||||
QString(QUrl::toPercentEncoding(entry->username())),
|
||||
QString(QUrl::toPercentEncoding(totp))));
|
||||
totp = url.toString(QUrl::FullyEncoded);
|
||||
}
|
||||
entry->setTotp(Totp::parseSettings(totp));
|
||||
}
|
||||
|
||||
if (loginMap.contains("itemEmail")) {
|
||||
entry->attributes()->set("login_email", loginMap.value("itemEmail").toString());
|
||||
}
|
||||
|
||||
// Set the entry url(s)
|
||||
int i = 1;
|
||||
for (const auto& urlObj : loginMap.value("urls").toList()) {
|
||||
const auto url = urlObj.toString();
|
||||
if (entry->url().isEmpty()) {
|
||||
// First url encountered is set as the primary url
|
||||
entry->setUrl(url);
|
||||
} else {
|
||||
// Subsequent urls
|
||||
entry->attributes()->set(
|
||||
QString("%1_%2").arg(EntryAttributes::AdditionalUrlAttribute, QString::number(i)), url);
|
||||
++i;
|
||||
}
|
||||
}
|
||||
}
|
||||
// Credit Card
|
||||
else if (type.compare("creditCard", Qt::CaseInsensitive) == 0) {
|
||||
const auto cardMap = dataMap.value("content").toMap();
|
||||
entry->setUsername(cardMap.value("number").toString());
|
||||
entry->setPassword(cardMap.value("verificationNumber").toString());
|
||||
const QStringList attrs({"cardholderName", "pin", "expirationDate"});
|
||||
const QStringList sensitive({"pin"});
|
||||
for (const auto& attr : attrs) {
|
||||
auto value = cardMap.value(attr).toString();
|
||||
if (!value.isEmpty()) {
|
||||
entry->attributes()->set("card_" + attr, value, sensitive.contains(attr));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Parse extra fields
|
||||
for (const auto& field : dataMap.value("extraFields").toList()) {
|
||||
// Derive a prefix for attribute names using the title or uuid if missing
|
||||
const auto fieldMap = field.toMap();
|
||||
auto name = fieldMap.value("fieldName").toString();
|
||||
if (entry->attributes()->hasKey(name)) {
|
||||
name = QString("%1_%2").arg(name, QUuid::createUuid().toString().mid(1, 5));
|
||||
}
|
||||
|
||||
QString value;
|
||||
const auto fieldType = fieldMap.value("type").toString();
|
||||
if (fieldType.compare("totp", Qt::CaseInsensitive) == 0) {
|
||||
value = fieldMap.value("data").toJsonObject().value("totpUri").toString();
|
||||
} else {
|
||||
value = fieldMap.value("data").toJsonObject().value("content").toString();
|
||||
}
|
||||
|
||||
entry->attributes()->set(name, value, fieldType.compare("hidden", Qt::CaseInsensitive) == 0);
|
||||
}
|
||||
|
||||
// Checked expired/deleted state
|
||||
if (itemMap.value("state").toInt() == 2) {
|
||||
entry->setExpires(true);
|
||||
entry->setExpiryTime(QDateTime::currentDateTimeUtc());
|
||||
}
|
||||
|
||||
// Collapse any accumulated history
|
||||
entry->removeHistoryItems(entry->historyItems());
|
||||
|
||||
// Adjust the created and modified times
|
||||
auto timeInfo = entry->timeInfo();
|
||||
const auto createdTime = QDateTime::fromSecsSinceEpoch(itemMap.value("createTime").toULongLong(), Qt::UTC);
|
||||
const auto modifiedTime = QDateTime::fromSecsSinceEpoch(itemMap.value("modifyTime").toULongLong(), Qt::UTC);
|
||||
timeInfo.setCreationTime(createdTime);
|
||||
timeInfo.setLastModificationTime(modifiedTime);
|
||||
timeInfo.setLastAccessTime(modifiedTime);
|
||||
entry->setTimeInfo(timeInfo);
|
||||
|
||||
return entry.take();
|
||||
}
|
||||
|
||||
void writeVaultToDatabase(const QJsonObject& vault, QSharedPointer<Database> db)
|
||||
{
|
||||
// Create groups from vaults and store a temporary map of id -> uuid
|
||||
const auto vaults = vault.value("vaults").toObject().toVariantMap();
|
||||
for (const auto& vaultId : vaults.keys()) {
|
||||
auto vaultObj = vaults.value(vaultId).toJsonObject();
|
||||
auto group = new Group();
|
||||
group->setUuid(QUuid::createUuid());
|
||||
group->setName(vaultObj.value("name").toString());
|
||||
group->setNotes(vaultObj.value("description").toString());
|
||||
group->setParent(db->rootGroup());
|
||||
|
||||
const auto items = vaultObj.value("items").toArray();
|
||||
for (const auto& item : items) {
|
||||
auto entry = readItem(item.toObject());
|
||||
if (entry) {
|
||||
entry->setGroup(group, false);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
} // namespace
|
||||
|
||||
bool ProtonPassReader::hasError()
|
||||
{
|
||||
return !m_error.isEmpty();
|
||||
}
|
||||
|
||||
QString ProtonPassReader::errorString()
|
||||
{
|
||||
return m_error;
|
||||
}
|
||||
|
||||
QSharedPointer<Database> ProtonPassReader::convert(const QString& path)
|
||||
{
|
||||
m_error.clear();
|
||||
|
||||
QFileInfo fileinfo(path);
|
||||
if (!fileinfo.exists()) {
|
||||
m_error = QObject::tr("File does not exist.").arg(path);
|
||||
return {};
|
||||
}
|
||||
|
||||
// Bitwarden uses a json file format
|
||||
QFile file(fileinfo.absoluteFilePath());
|
||||
if (!file.open(QFile::ReadOnly)) {
|
||||
m_error = QObject::tr("Cannot open file: %1").arg(file.errorString());
|
||||
return {};
|
||||
}
|
||||
|
||||
QJsonParseError error;
|
||||
auto json = QJsonDocument::fromJson(file.readAll(), &error).object();
|
||||
if (error.error != QJsonParseError::NoError) {
|
||||
m_error =
|
||||
QObject::tr("Cannot parse file: %1 at position %2").arg(error.errorString(), QString::number(error.offset));
|
||||
return {};
|
||||
}
|
||||
|
||||
file.close();
|
||||
|
||||
if (json.value("encrypted").toBool()) {
|
||||
m_error = QObject::tr("Encrypted files are not supported.");
|
||||
return {};
|
||||
}
|
||||
|
||||
auto db = QSharedPointer<Database>::create();
|
||||
db->rootGroup()->setName(QObject::tr("Proton Pass Import"));
|
||||
|
||||
writeVaultToDatabase(json, db);
|
||||
|
||||
return db;
|
||||
}
|
43
src/format/ProtonPassReader.h
Normal file
43
src/format/ProtonPassReader.h
Normal file
|
@ -0,0 +1,43 @@
|
|||
/*
|
||||
* Copyright (C) 2024 KeePassXC Team <team@keepassxc.org>
|
||||
*
|
||||
* 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 <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#ifndef PROTONPASS_READER_H
|
||||
#define PROTONPASS_READER_H
|
||||
|
||||
#include <QSharedPointer>
|
||||
|
||||
class Database;
|
||||
|
||||
/*!
|
||||
* Imports a Proton Pass vault in JSON format: https://proton.me/support/pass-export
|
||||
*/
|
||||
class ProtonPassReader
|
||||
{
|
||||
public:
|
||||
explicit ProtonPassReader() = default;
|
||||
~ProtonPassReader() = default;
|
||||
|
||||
QSharedPointer<Database> convert(const QString& path);
|
||||
|
||||
bool hasError();
|
||||
QString errorString();
|
||||
|
||||
private:
|
||||
QString m_error;
|
||||
};
|
||||
|
||||
#endif // PROTONPASS_READER_H
|
|
@ -303,6 +303,9 @@ void DatabaseTabWidget::importFile()
|
|||
Merger merger(db.data(), newDb.data());
|
||||
merger.setSkipDatabaseCustomData(true);
|
||||
merger.merge();
|
||||
// Transfer the root group data
|
||||
newDb->rootGroup()->setName(db->rootGroup()->name());
|
||||
newDb->rootGroup()->setNotes(db->rootGroup()->notes());
|
||||
// Show the new database
|
||||
auto dbWidget = new DatabaseWidget(newDb, this);
|
||||
addDatabaseTab(dbWidget);
|
||||
|
|
|
@ -48,6 +48,7 @@ public:
|
|||
IMPORT_OPVAULT,
|
||||
IMPORT_OPUX,
|
||||
IMPORT_BITWARDEN,
|
||||
IMPORT_PROTONPASS,
|
||||
IMPORT_KEEPASS1,
|
||||
IMPORT_REMOTE,
|
||||
};
|
||||
|
|
|
@ -24,6 +24,7 @@
|
|||
#include "format/KeePass1Reader.h"
|
||||
#include "format/OPUXReader.h"
|
||||
#include "format/OpVaultReader.h"
|
||||
#include "format/ProtonPassReader.h"
|
||||
#include "gui/csvImport/CsvImportWidget.h"
|
||||
#include "gui/wizard/ImportWizard.h"
|
||||
|
||||
|
@ -75,34 +76,35 @@ void ImportWizardPageReview::initializePage()
|
|||
break;
|
||||
case ImportWizard::IMPORT_OPVAULT:
|
||||
m_db = importOPVault(filename, field("ImportPassword").toString());
|
||||
setupDatabasePreview();
|
||||
break;
|
||||
case ImportWizard::IMPORT_OPUX:
|
||||
m_db = importOPUX(filename);
|
||||
setupDatabasePreview();
|
||||
break;
|
||||
case ImportWizard::IMPORT_KEEPASS1:
|
||||
m_db = importKeePass1(filename, field("ImportPassword").toString(), field("ImportKeyFile").toString());
|
||||
setupDatabasePreview();
|
||||
break;
|
||||
case ImportWizard::IMPORT_BITWARDEN:
|
||||
m_db = importBitwarden(filename, field("ImportPassword").toString());
|
||||
setupDatabasePreview();
|
||||
break;
|
||||
case ImportWizard::IMPORT_PROTONPASS:
|
||||
m_db = importProtonPass(filename);
|
||||
break;
|
||||
case ImportWizard::IMPORT_REMOTE:
|
||||
m_db = importRemote(field("DownloadCommand").toString(),
|
||||
field("DownloadInput").toString(),
|
||||
field("ImportPassword").toString(),
|
||||
field("ImportKeyFile").toString());
|
||||
setupDatabasePreview();
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
setupDatabasePreview();
|
||||
}
|
||||
|
||||
bool ImportWizardPageReview::validatePage()
|
||||
{
|
||||
if (m_csvWidget && field("ImportType").toInt() == ImportWizard::IMPORT_CSV) {
|
||||
if (isCsvImport()) {
|
||||
m_db = m_csvWidget->buildDatabase();
|
||||
}
|
||||
return !m_db.isNull();
|
||||
|
@ -124,14 +126,18 @@ void ImportWizardPageReview::setupCsvImport(const QString& filename)
|
|||
});
|
||||
|
||||
m_csvWidget->load(filename);
|
||||
|
||||
// Qt does not automatically resize a QScrollWidget in a QWizard...
|
||||
m_ui->scrollAreaContents->layout()->addWidget(m_csvWidget);
|
||||
m_ui->scrollArea->setMinimumSize(m_csvWidget->width() + 50, m_csvWidget->height() + 100);
|
||||
}
|
||||
|
||||
void ImportWizardPageReview::setupDatabasePreview()
|
||||
{
|
||||
// CSV preview is handled by the import widget
|
||||
if (isCsvImport()) {
|
||||
// Qt does not automatically resize a QScrollWidget in a QWizard...
|
||||
m_ui->scrollAreaContents->layout()->addWidget(m_csvWidget);
|
||||
m_ui->scrollArea->setMinimumSize(m_csvWidget->width() + 50, m_csvWidget->height() + 100);
|
||||
return;
|
||||
}
|
||||
|
||||
if (!m_db) {
|
||||
m_ui->scrollArea->setVisible(false);
|
||||
return;
|
||||
|
@ -216,6 +222,21 @@ ImportWizardPageReview::importKeePass1(const QString& filename, const QString& p
|
|||
return db;
|
||||
}
|
||||
|
||||
QSharedPointer<Database> ImportWizardPageReview::importProtonPass(const QString& filename)
|
||||
{
|
||||
ProtonPassReader reader;
|
||||
auto db = reader.convert(filename);
|
||||
if (reader.hasError()) {
|
||||
m_ui->messageWidget->showMessage(reader.errorString(), KMessageWidget::Error, -1);
|
||||
}
|
||||
return db;
|
||||
}
|
||||
|
||||
bool ImportWizardPageReview::isCsvImport() const
|
||||
{
|
||||
return m_csvWidget && field("ImportType").toInt() == ImportWizard::IMPORT_CSV;
|
||||
}
|
||||
|
||||
QSharedPointer<Database> ImportWizardPageReview::importRemote(const QString& downloadCommand,
|
||||
const QString& downloadInput,
|
||||
const QString& password,
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
/*
|
||||
/*
|
||||
* Copyright (C) 2023 KeePassXC Team <team@keepassxc.org>
|
||||
*
|
||||
* This program is free software: you can redistribute it and/or modify
|
||||
|
@ -49,11 +49,13 @@ public:
|
|||
QSharedPointer<Database> database();
|
||||
|
||||
private:
|
||||
bool isCsvImport() const;
|
||||
void setupCsvImport(const QString& filename);
|
||||
QSharedPointer<Database> importOPUX(const QString& filename);
|
||||
QSharedPointer<Database> importBitwarden(const QString& filename, const QString& password);
|
||||
QSharedPointer<Database> importOPVault(const QString& filename, const QString& password);
|
||||
QSharedPointer<Database> importKeePass1(const QString& filename, const QString& password, const QString& keyfile);
|
||||
QSharedPointer<Database> importProtonPass(const QString& filename);
|
||||
QSharedPointer<Database> importRemote(const QString& downloadCommand,
|
||||
const QString& downloadInput,
|
||||
const QString& password,
|
||||
|
|
|
@ -37,15 +37,17 @@ ImportWizardPageSelect::ImportWizardPageSelect(QWidget* parent)
|
|||
new QListWidgetItem(icons()->icon("onepassword"), tr("1Password Export (.1pux)"), m_ui->importTypeList);
|
||||
new QListWidgetItem(icons()->icon("onepassword"), tr("1Password Vault (.opvault)"), m_ui->importTypeList);
|
||||
new QListWidgetItem(icons()->icon("bitwarden"), tr("Bitwarden (.json)"), m_ui->importTypeList);
|
||||
new QListWidgetItem(icons()->icon("object-locked"), tr("KeePass 1 Database (.kdb)"), m_ui->importTypeList);
|
||||
new QListWidgetItem(icons()->icon("proton"), tr("Proton Pass (.json)"), m_ui->importTypeList);
|
||||
new QListWidgetItem(icons()->icon("web"), tr("Remote Database (.kdbx)"), m_ui->importTypeList);
|
||||
new QListWidgetItem(icons()->icon("object-locked"), tr("KeePass 1 Database (.kdb)"), m_ui->importTypeList);
|
||||
|
||||
m_ui->importTypeList->item(0)->setData(Qt::UserRole, ImportWizard::IMPORT_CSV);
|
||||
m_ui->importTypeList->item(1)->setData(Qt::UserRole, ImportWizard::IMPORT_OPUX);
|
||||
m_ui->importTypeList->item(2)->setData(Qt::UserRole, ImportWizard::IMPORT_OPVAULT);
|
||||
m_ui->importTypeList->item(3)->setData(Qt::UserRole, ImportWizard::IMPORT_BITWARDEN);
|
||||
m_ui->importTypeList->item(4)->setData(Qt::UserRole, ImportWizard::IMPORT_KEEPASS1);
|
||||
m_ui->importTypeList->item(4)->setData(Qt::UserRole, ImportWizard::IMPORT_PROTONPASS);
|
||||
m_ui->importTypeList->item(5)->setData(Qt::UserRole, ImportWizard::IMPORT_REMOTE);
|
||||
m_ui->importTypeList->item(6)->setData(Qt::UserRole, ImportWizard::IMPORT_KEEPASS1);
|
||||
|
||||
connect(m_ui->importTypeList, &QListWidget::currentItemChanged, this, &ImportWizardPageSelect::itemSelected);
|
||||
m_ui->importTypeList->setCurrentRow(0);
|
||||
|
@ -132,6 +134,7 @@ void ImportWizardPageSelect::itemSelected(QListWidgetItem* current, QListWidgetI
|
|||
// Unencrypted types
|
||||
case ImportWizard::IMPORT_CSV:
|
||||
case ImportWizard::IMPORT_OPUX:
|
||||
case ImportWizard::IMPORT_PROTONPASS:
|
||||
setCredentialState(false);
|
||||
setDownloadCommand(false);
|
||||
break;
|
||||
|
@ -299,6 +302,8 @@ QString ImportWizardPageSelect::importFileFilter()
|
|||
return QString("%1 (*.1pux)").arg(tr("1Password Export"));
|
||||
case ImportWizard::IMPORT_BITWARDEN:
|
||||
return QString("%1 (*.json)").arg(tr("Bitwarden JSON Export"));
|
||||
case ImportWizard::IMPORT_PROTONPASS:
|
||||
return QString("%1 (*.json)").arg(tr("Proton Pass JSON Export"));
|
||||
case ImportWizard::IMPORT_OPVAULT:
|
||||
return QString("%1 (*.opvault)").arg(tr("1Password Vault"));
|
||||
case ImportWizard::IMPORT_KEEPASS1:
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue