8#include "kwalletbackend.h"
9#include "kwalletbackend_debug.h"
15#include <gpgme++/key.h>
18#include <KNotification>
19#include <KLocalizedString>
25#include <QCryptographicHash>
26#include <QRegularExpression>
27#include <QStandardPaths>
42#define KWALLETSTORAGE_VERSION_MAJOR 0
43#define KWALLETSTORAGE_VERSION_MINOR 1
45using namespace KWallet;
47#define KWMAGIC "KWALLET\n\r\0\r\n"
49static const QByteArray walletAllowedChars =
"abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789^&'@{}[],$=!-#()%.+_\r\n\t\f\v ";
61QString Backend::decodeWalletName(
const QString &encodedName) {
65gcry_error_t ensureGcryptInit()
67 bool static gcry_secmem_init =
false;
68 if (gcry_secmem_init) {
71 gcry_error_t
error = gcry_control(GCRYCTL_INIT_SECMEM, 32768, 0);
73 qCWarning(KWALLETBACKEND_LOG) <<
"Can't get secure memory:" <<
error;
76 gcry_secmem_init =
true;
78 gcry_control(GCRYCTL_INITIALIZATION_FINISHED, 0);
83class Backend::BackendPrivate
92Backend::Backend(
const QString &name,
bool isPath)
95 _cipherType(KWallet::BACKEND_CIPHER_UNKNOWN)
104 _path = getSaveLocation() +
'/' + encodeWalletName(_name) +
".kwl";
118QString Backend::getSaveLocation()
121 QDir writeDir(writeLocation);
122 if (!writeDir.exists()) {
123 if (!writeDir.mkpath(writeLocation)) {
124 qFatal(
"Cannot create wallet save location!");
129 return writeLocation;
132void Backend::setCipherType(BackendCipherType ct)
135 assert(_cipherType == KWallet::BACKEND_CIPHER_UNKNOWN);
139static int password2PBKDF2_SHA512(
const QByteArray &password, QByteArray &hash,
const QByteArray &salt)
141 if (!gcry_check_version(
"1.5.0")) {
142 qCWarning(KWALLETBACKEND_LOG) <<
"libcrypt version is too old";
143 return GPG_ERR_USER_2;
146 gcry_error_t
error = ensureGcryptInit();
152 GCRY_KDF_PBKDF2, GCRY_MD_SHA512,
154 PBKDF2_SHA512_ITERATIONS, PBKDF2_SHA512_KEYSIZE, hash.
data());
160static int password2hash(
const QByteArray &password, QByteArray &hash)
163 int shasz = sha.size() / 8;
167 QByteArray block1(shasz, 0);
169 sha.process(password.
data(), qMin(password.
size(), 16));
172 for (
int i = 0; i < 2000; i++) {
173 memcpy(block1.data(), sha.hash(), shasz);
175 sha.process(block1.data(), shasz);
180 if (password.
size() > 16) {
181 sha.process(password.
data() + 16, qMin(password.
size() - 16, 16));
182 QByteArray block2(shasz, 0);
184 for (
int i = 0; i < 2000; i++) {
185 memcpy(block2.data(), sha.hash(), shasz);
187 sha.process(block2.data(), shasz);
192 if (password.
size() > 32) {
193 sha.process(password.
data() + 32, qMin(password.
size() - 32, 16));
195 QByteArray block3(shasz, 0);
197 for (
int i = 0; i < 2000; i++) {
198 memcpy(block3.data(), sha.hash(), shasz);
200 sha.process(block3.data(), shasz);
205 if (password.
size() > 48) {
206 sha.process(password.
data() + 48, password.
size() - 48);
208 QByteArray block4(shasz, 0);
210 for (
int i = 0; i < 2000; i++) {
211 memcpy(block4.data(), sha.hash(), shasz);
213 sha.process(block4.data(), shasz);
219 memcpy(hash.
data(), block1.data(), 14);
220 memcpy(hash.
data() + 14, block2.data(), 14);
221 memcpy(hash.
data() + 28, block3.data(), 14);
222 memcpy(hash.
data() + 42, block4.data(), 14);
227 memcpy(hash.
data(), block1.data(), 20);
228 memcpy(hash.
data() + 20, block2.data(), 20);
229 memcpy(hash.
data() + 40, block3.data(), 16);
235 memcpy(hash.
data(), block1.data(), 20);
236 memcpy(hash.
data() + 20, block2.data(), 20);
242 memcpy(hash.
data(), block1.data(), 20);
253 qCDebug(KWALLETBACKEND_LOG) <<
"refCount negative!";
259bool Backend::exists(
const QString &wallet)
261 QString saveLocation = getSaveLocation();
262 QString
path = saveLocation +
'/' + encodeWalletName(wallet) + QLatin1String(
".kwl");
268QString Backend::openRCToString(
int rc)
272 return i18n(
"Already open.");
274 return i18n(
"Error opening file.");
276 return i18n(
"Not a wallet file.");
278 return i18n(
"Unsupported file format revision.");
280 return QStringLiteral(
"Unknown cipher or hash");
282 return i18n(
"Unknown encryption scheme.");
284 return i18n(
"Corrupt file?");
286 return i18n(
"Error validating wallet integrity. Possibly corrupted.");
290 return i18n(
"Read error - possibly incorrect password.");
292 return i18n(
"Decryption error.");
298int Backend::open(
const QByteArray &password, WId w)
304 setPassword(password);
305 return openInternal(w);
309int Backend::open(
const GpgME::Key &key)
315 return openInternal();
319int Backend::openPreHashed(
const QByteArray &passwordHash)
326 if (passwordHash.
size() != 20 && passwordHash.
size() != 40 &&
327 passwordHash.
size() != 56) {
331 _passhash = passwordHash;
332 _newPassHash = passwordHash;
335 return openInternal();
338int Backend::openInternal(WId w)
344 QFile newfile(_path);
361 char magicBuf[KWMAGIC_LEN];
362 db.read(magicBuf, KWMAGIC_LEN);
363 if (memcmp(magicBuf, KWMAGIC, KWMAGIC_LEN) != 0) {
367 db.read(magicBuf, 4);
370 if (magicBuf[0] != KWALLETSTORAGE_VERSION_MAJOR) {
375 if (magicBuf[1] == 1) {
376 qCDebug(KWALLETBACKEND_LOG) <<
"Wallet new enough, using new hash";
378 }
else if (magicBuf[1] != 0) {
379 qCDebug(KWALLETBACKEND_LOG) <<
"Wallet is old, sad panda :(";
383 BackendPersistHandler *phandler = BackendPersistHandler::getPersistHandler(magicBuf);
384 if (
nullptr == phandler) {
387 int result = phandler->read(
this, db, w);
392void Backend::swapToNewHash()
396 qCDebug(KWALLETBACKEND_LOG) <<
"Runtime error on the new hash";
400 _passhash = _newPassHash;
403QByteArray Backend::createAndSaveSalt(
const QString &path)
const
405 QFile saltFile(path);
413 if (ensureGcryptInit() != 0) {
417 QByteArray salt(PBKDF2_SHA512_SALTSIZE, Qt::Initialization::Uninitialized);
418 gcry_randomize(salt.
data(), salt.
size(), GCRY_STRONG_RANDOM);
421 if (saltFile.write(salt) != PBKDF2_SHA512_SALTSIZE) {
430int Backend::sync(WId w)
447 if (sf.write(KWMAGIC, KWMAGIC_LEN) != KWMAGIC_LEN) {
454 version[0] = KWALLETSTORAGE_VERSION_MAJOR;
456 version[1] = KWALLETSTORAGE_VERSION_MINOR;
463 BackendPersistHandler *phandler = BackendPersistHandler::getPersistHandler(_cipherType);
464 if (
nullptr == phandler) {
467 int rc = phandler->write(
this, sf, version, w);
471 KNotification *notification =
new KNotification(QStringLiteral(
"syncFailed"));
472 notification->
setText(
i18n(
"Failed to sync wallet <b>%1</b> to disk. Error codes are:\nRC <b>%2</b>\nSF <b>%3</b>. Please file a BUG report using this information to bugs.kde.org", _name, rc, sf.errorString()));
479int Backend::closeInternal(
bool save)
501int Backend::close(
bool save)
503 int rc = closeInternal(save);
509 _newPassHash.fill(0);
514const QString &Backend::walletName()
const
519int Backend::renameWallet(
const QString &newName,
bool isPath)
522 const auto saveLocation = getSaveLocation();
527 newPath = saveLocation +
QChar::fromLatin1(
'/') + encodeWalletName(newName) + QStringLiteral(
".kwl");
530 if (newPath == _path) {
538 int rc = closeInternal(
true);
545 saveLocation +
QChar::fromLatin1(
'/') + encodeWalletName(newName) + QStringLiteral(
".salt"));
558bool Backend::isOpen()
const
563QStringList Backend::folderList()
const
565 return _entries.keys();
568QStringList Backend::entryList()
const
570 return _entries[_folder].keys();
573Entry *Backend::readEntry(
const QString &key)
577 if (_open && hasEntry(key)) {
578 rc = _entries[_folder][key];
584#if KWALLET_BUILD_DEPRECATED_SINCE(5, 72)
585QList<Entry *> Backend::readEntryList(
const QString &key)
595 QLatin1String(
"[^/]"), QLatin1String(
"."));
596 const QRegularExpression re(pattern);
598 const EntryMap &
map = _entries[_folder];
600 if (re.match(i.key()).hasMatch()) {
608QList<Entry *> Backend::entriesList()
const
611 return QList<Entry *>();
613 const EntryMap &
map = _entries[_folder];
619bool Backend::createFolder(
const QString &f)
621 if (_entries.contains(f)) {
625 _entries.insert(f, EntryMap());
628 folderMd5.addData(f.
toUtf8());
629 _hashes.insert(MD5Digest(folderMd5.result()), QList<MD5Digest>());
634int Backend::renameEntry(
const QString &oldName,
const QString &newName)
636 EntryMap &emap = _entries[_folder];
640 if (oi != emap.end() && ni == emap.end()) {
641 Entry *e = oi.value();
646 folderMd5.addData(_folder.toUtf8());
648 HashMap::iterator i = _hashes.find(MD5Digest(folderMd5.result()));
649 if (i != _hashes.end()) {
652 oldMd5.addData(oldName.
toUtf8());
653 newMd5.addData(newName.
toUtf8());
654 i.value().removeAll(MD5Digest(oldMd5.result()));
655 i.value().append(MD5Digest(newMd5.result()));
663void Backend::writeEntry(Entry *e)
669 if (!hasEntry(e->key())) {
670 _entries[_folder][e->key()] =
new Entry;
672 _entries[_folder][e->key()]->copy(e);
675 folderMd5.addData(_folder.toUtf8());
677 HashMap::iterator i = _hashes.find(MD5Digest(folderMd5.result()));
678 if (i != _hashes.end()) {
680 md5.addData(e->key().
toUtf8());
681 i.value().append(MD5Digest(md5.result()));
685bool Backend::hasEntry(
const QString &key)
const
687 return _entries.contains(_folder) && _entries[_folder].contains(key);
690bool Backend::removeEntry(
const QString &key)
699 if (fi != _entries.end() && ei != fi.value().end()) {
701 fi.value().erase(ei);
703 folderMd5.addData(_folder.toUtf8());
705 HashMap::iterator i = _hashes.find(MD5Digest(folderMd5.result()));
706 if (i != _hashes.end()) {
708 md5.addData(key.
toUtf8());
709 i.value().removeAll(MD5Digest(md5.result()));
717bool Backend::removeFolder(
const QString &f)
725 if (fi != _entries.end()) {
737 folderMd5.addData(f.
toUtf8());
738 _hashes.remove(MD5Digest(folderMd5.result()));
745bool Backend::folderDoesNotExist(
const QString &folder)
const
748 md5.addData(folder.toUtf8());
749 return !_hashes.contains(MD5Digest(md5.result()));
752bool Backend::entryDoesNotExist(
const QString &folder,
const QString &entry)
const
755 md5.addData(folder.toUtf8());
756 HashMap::const_iterator i = _hashes.find(MD5Digest(md5.result()));
757 if (i != _hashes.end()) {
759 md5.addData(entry.
toUtf8());
760 return !i.value().contains(MD5Digest(md5.result()));
765void Backend::setPassword(
const QByteArray &password)
769 CipherBlockChain bf(&_bf);
770 _passhash.resize(bf.keyLen() / 8);
771 _newPassHash.resize(bf.keyLen() / 8);
772 _newPassHash.fill(0);
774 password2hash(password, _passhash);
777 QFile saltFile(getSaveLocation() +
'/' + encodeWalletName(_name) +
".salt");
778 if (!saltFile.exists() || saltFile.size() == 0) {
779 salt = createAndSaveSalt(saltFile.fileName());
782 salt = createAndSaveSalt(saltFile.fileName());
784 salt = saltFile.readAll();
788 if (!salt.
isEmpty() && password2PBKDF2_SHA512(password, _newPassHash, salt) == 0) {
789 qCDebug(KWALLETBACKEND_LOG) <<
"Setting useNewHash to true";
795const GpgME::Key &Backend::gpgKey()
const
void setText(const QString &text)
QString i18n(const char *text, const TYPE &arg...)
KDB_EXPORT KDbVersionInfo version()
QString path(const QString &relativePath)
void error(QWidget *parent, const QString &text, const QString &title, const KGuiItem &buttonOk, Options options=Notify)
QString name(StandardAction id)
const char * constData() const const
QByteArray fromPercentEncoding(const QByteArray &input, char percent)
bool isEmpty() const const
void resize(qsizetype newSize, char c)
qsizetype size() const const
QByteArray toPercentEncoding(const QByteArray &exclude, const QByteArray &include, char percent) const const
bool exists() const const
bool rename(const QString &newName)
void append(QList< T > &&value)
QString wildcardToRegularExpression(QStringView pattern, WildcardConversionOptions options)
QString writableLocation(StandardLocation type)
QString fromUtf8(QByteArrayView str)
QString & replace(QChar before, QChar after, Qt::CaseSensitivity cs)
QByteArray toUtf8() const const
QFuture< void > map(Iterator begin, Iterator end, MapFunctor &&function)