You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tdelibs/tdeio/misc/tdewalletd/tdewalletd.cpp

1515 lines
39 KiB

/*
This file is part of the KDE libraries
Copyright (c) 2002-2004 George Staikos <staikos@kde.org>
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Library General Public
License as published by the Free Software Foundation; either
version 2 of the License, or (at your option) any later version.
This library 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
Library General Public License for more details.
You should have received a copy of the GNU Library General Public License
along with this library; see the file COPYING.LIB. If not, write to
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
Boston, MA 02110-1301, USA.
*/
#include "kbetterthankdialogbase.h"
#include "tdewalletwizard.h"
#include "tdewalletd.h"
#include "ktimeout.h"
#include <dcopclient.h>
#include <dcopref.h>
#include <kactivelabel.h>
#include <tdeapplication.h>
#include <tdeconfig.h>
#include <kdebug.h>
#include <kdirwatch.h>
#include <tdeglobal.h>
#include <tdelocale.h>
#include <tdemessagebox.h>
#include <kpassdlg.h>
#include <kstandarddirs.h>
#include <tdewalletentry.h>
#include <twin.h>
#include <tqdir.h>
#include <tqlabel.h>
#include <tqlayout.h>
#include <tqpushbutton.h>
#include <tqregexp.h>
#include <tqstylesheet.h>
#include <tqvbox.h>
#include <assert.h>
extern "C" {
KDE_EXPORT KDEDModule *create_tdewalletd(const TQCString &name) {
return new TDEWalletD(name);
}
}
class TDEWalletTransaction {
public:
TDEWalletTransaction() {
tType = Unknown;
transaction = 0L;
client = 0L;
modal = false;
}
~TDEWalletTransaction() {
// Don't delete these!
transaction = 0L;
client = 0L;
}
enum Type { Unknown, Open, ChangePassword, OpenFail };
DCOPClient *client;
DCOPClientTransaction *transaction;
Type tType;
TQCString rawappid, returnObject;
TQCString appid;
uint wId;
TQString wallet;
bool modal;
};
TDEWalletD::TDEWalletD(const TQCString &name)
: KDEDModule(name), _failed(0) {
srand(time(0));
_showingFailureNotify = false;
_transactions.setAutoDelete(true);
_timeouts = new KTimeout(17);
_closeIdle = false;
_idleTime = 0;
connect(_timeouts, TQT_SIGNAL(timedOut(int)), this, TQT_SLOT(timedOut(int)));
reconfigure();
TDEGlobal::dirs()->addResourceType("tdewallet", "share/apps/tdewallet");
connect(TDEApplication::dcopClient(),
TQT_SIGNAL(applicationRemoved(const TQCString&)),
this,
TQT_SLOT(slotAppUnregistered(const TQCString&)));
_dw = new KDirWatch(this, "TDEWallet Directory Watcher");
_dw->addDir(TDEGlobal::dirs()->saveLocation("tdewallet"));
_dw->startScan(true);
connect(_dw, TQT_SIGNAL(dirty(const TQString&)), this, TQT_SLOT(emitWalletListDirty()));
}
TDEWalletD::~TDEWalletD() {
delete _timeouts;
_timeouts = 0;
closeAllWallets();
_transactions.clear();
}
int TDEWalletD::generateHandle() {
int rc;
// ASSUMPTION: RAND_MAX is fairly large.
do {
rc = rand();
} while (_wallets.find(rc) || rc == 0);
return rc;
}
void TDEWalletD::processTransactions() {
static bool processing = false;
if (processing) {
return;
}
processing = true;
// Process remaining transactions
TDEWalletTransaction *xact;
while (!_transactions.isEmpty()) {
xact = _transactions.first();
TQCString replyType;
int res;
assert(xact->tType != TDEWalletTransaction::Unknown);
switch (xact->tType) {
case TDEWalletTransaction::Open:
res = doTransactionOpen(xact->appid, xact->wallet, xact->wId, xact->modal);
replyType = "int";
if (!xact->returnObject.isEmpty()) {
DCOPRef(xact->rawappid, xact->returnObject).send("walletOpenResult", res);
}
// multiple requests from the same client
// should not produce multiple password
// dialogs on a failure
if (res < 0) {
TQPtrListIterator<TDEWalletTransaction> it(_transactions);
TDEWalletTransaction *x;
while ((x = it.current()) && x != xact) {
++it;
}
if (x) {
++it;
}
while ((x = it.current())) {
if (xact->appid == x->appid && x->tType == TDEWalletTransaction::Open && x->wallet == xact->wallet && x->wId == xact->wId) {
x->tType = TDEWalletTransaction::OpenFail;
}
++it;
}
}
break;
case TDEWalletTransaction::OpenFail:
res = -1;
replyType = "int";
if (!xact->returnObject.isEmpty()) {
DCOPRef(xact->rawappid, xact->returnObject).send("walletOpenResult", res);
}
break;
case TDEWalletTransaction::ChangePassword:
doTransactionChangePassword(xact->appid, xact->wallet, xact->wId);
// fall through - no return
default:
_transactions.removeRef(xact);
continue;
}
if (xact->returnObject.isEmpty() && xact->tType != TDEWalletTransaction::ChangePassword) {
TQByteArray replyData;
TQDataStream stream(replyData, IO_WriteOnly);
stream << res;
xact->client->endTransaction(xact->transaction, replyType, replyData);
}
_transactions.removeRef(xact);
}
processing = false;
}
void TDEWalletD::openAsynchronous(const TQString& wallet, const TQCString& returnObject, uint wId) {
DCOPClient *dc = callingDcopClient();
if (!dc) {
return;
}
TQCString appid = dc->senderId();
if (!_enabled ||
!TQRegExp("^[A-Za-z0-9]+[A-Za-z0-9\\s\\-_]*$").exactMatch(wallet)) {
DCOPRef(appid, returnObject).send("walletOpenResult", -1);
return;
}
TQCString peerName = friendlyDCOPPeerName();
TDEWalletTransaction *xact = new TDEWalletTransaction;
xact->appid = peerName;
xact->rawappid = appid;
xact->client = callingDcopClient();
xact->wallet = wallet;
xact->wId = wId;
xact->tType = TDEWalletTransaction::Open;
xact->returnObject = returnObject;
_transactions.append(xact);
DCOPRef(appid, returnObject).send("walletOpenResult", 0);
TQTimer::singleShot(0, this, TQT_SLOT(processTransactions()));
checkActiveDialog();
}
int TDEWalletD::openPath(const TQString& path, uint wId) {
if (!_enabled) { // guard
return -1;
}
// FIXME: setup transaction
int rc = internalOpen(friendlyDCOPPeerName(), path, true, wId);
return rc;
}
int TDEWalletD::open(const TQString& wallet, uint wId) {
if (!_enabled) { // guard
return -1;
}
if (!TQRegExp("^[A-Za-z0-9]+[A-Za-z0-9\\s\\-_]*$").exactMatch(wallet)) {
return -1;
}
TQCString appid = friendlyDCOPPeerName();
TDEWalletTransaction *xact = new TDEWalletTransaction;
_transactions.append(xact);
xact->appid = appid;
xact->client = callingDcopClient();
xact->transaction = xact->client->beginTransaction();
xact->wallet = wallet;
xact->wId = wId;
xact->tType = TDEWalletTransaction::Open;
xact->modal = true; // mark dialogs as modal, the app has blocking wait
TQTimer::singleShot(0, this, TQT_SLOT(processTransactions()));
checkActiveDialog();
return 0; // process later
}
// Sets up a dialog that will be shown by tdewallet.
void TDEWalletD::setupDialog( TQWidget* dialog, WId wId, const TQCString& appid, bool modal ) {
if( wId != 0 )
KWin::setMainWindow( dialog, wId ); // correct, set dialog parent
else {
if( appid.isEmpty())
kdWarning() << "Using tdewallet without parent window!" << endl;
else
kdWarning() << "Application '" << appid << "' using tdewallet without parent window!" << endl;
// allow dialog activation even if it interrupts, better than trying hacks
// with keeping the dialog on top or on all desktops
kapp->updateUserTimestamp();
}
if( modal )
KWin::setState( dialog->winId(), NET::Modal );
else
KWin::clearState( dialog->winId(), NET::Modal );
activeDialog = dialog;
}
// If there's a dialog already open and another application tries some operation that'd lead to
// opening a dialog, that application will be blocked by this dialog. A proper solution would
// be to set the second application's window also as a parent for the active dialog, so that
// KWin properly handles focus changes and so on, but there's currently no support for multiple
// dialog parents. Hopefully to be done in KDE4, for now just use all kinds of bad hacks to make
// sure the user doesn't overlook the active dialog.
void TDEWalletD::checkActiveDialog() {
if( !activeDialog || !activeDialog->isShown())
return;
kapp->updateUserTimestamp();
KWin::setState( activeDialog->winId(), NET::KeepAbove );
KWin::setOnAllDesktops( activeDialog->winId(), true );
KWin::forceActiveWindow( activeDialog->winId());
}
int TDEWalletD::doTransactionOpen(const TQCString& appid, const TQString& wallet, uint wId, bool modal) {
if (_firstUse && !wallets().contains(TDEWallet::Wallet::LocalWallet())) {
// First use wizard
TDEWalletWizard *wiz = new TDEWalletWizard(0);
setupDialog( wiz, wId, appid, modal );
int rc = wiz->exec();
if (rc == TQDialog::Accepted) {
TDEConfig cfg("tdewalletrc");
cfg.setGroup("Wallet");
cfg.writeEntry("First Use", false);
cfg.writeEntry("Enabled", wiz->_useWallet->isChecked());
cfg.writeEntry("Close When Idle", wiz->_closeIdle->isChecked());
cfg.writeEntry("Use One Wallet", !wiz->_networkWallet->isChecked());
cfg.sync();
reconfigure();
if (!wiz->_useWallet->isChecked()) {
delete wiz;
return -1;
}
// Create the wallet
TDEWallet::Backend *b = new TDEWallet::Backend(TDEWallet::Wallet::LocalWallet());
TQByteArray p;
p.duplicate(wiz->_pass1->text().utf8(), wiz->_pass1->text().length());
b->open(p);
b->createFolder(TDEWallet::Wallet::PasswordFolder());
b->createFolder(TDEWallet::Wallet::FormDataFolder());
b->close(p);
p.fill(0);
delete b;
delete wiz;
} else {
delete wiz;
return -1;
}
} else if (_firstUse) {
TDEConfig cfg("tdewalletrc");
_firstUse = false;
cfg.setGroup("Wallet");
cfg.writeEntry("First Use", false);
cfg.sync();
}
int rc = internalOpen(appid, wallet, false, wId, modal);
return rc;
}
int TDEWalletD::tryOpen(const TQString& wallet, const TQCString& password)
{
if (isOpen(wallet))
return 0;
if (_tryOpenBlocked.isActive()) {
kdDebug() << "tryOpen is active.." << endl;
return -1;
}
if (!TDEWallet::Backend::exists(wallet))
return -2;
TDEWallet::Backend *b = new TDEWallet::Backend(wallet, false /*isPath*/);
int rc = b->open(TQByteArray().duplicate(password, strlen(password)));
if (rc == 0) {
_wallets.insert(rc = generateHandle(), b);
_passwords[wallet] = password;
b->ref();
_tryOpenBlocked.stop();
TQByteArray data;
TQDataStream ds(data, IO_WriteOnly);
ds << wallet;
emitDCOPSignal("walletOpened(TQString)", data);
}
else {
delete b;
// make sure that we're not bombed with a dictionary attack
_tryOpenBlocked.start (30 * 1000, true /*single shot*/);
if (++_failed > 5) {
_failed = 0;
TQTimer::singleShot(0, this, TQT_SLOT(notifyFailures()));
}
rc = -1;
}
return rc;
}
int TDEWalletD::internalOpen(const TQCString& appid, const TQString& wallet, bool isPath, WId w, bool modal) {
int rc = -1;
bool brandNew = false;
TQCString thisApp;
if (appid.isEmpty()) {
thisApp = "TDE System";
} else {
thisApp = appid;
}
if (implicitDeny(wallet, thisApp)) {
return -1;
}
for (TQIntDictIterator<TDEWallet::Backend> i(_wallets); i.current(); ++i) {
if (i.current()->walletName() == wallet) {
rc = i.currentKey();
break;
}
}
if (rc == -1) {
if (_wallets.count() > 20) {
kdDebug() << "Too many wallets open." << endl;
return -1;
}
TDEWallet::Backend *b = new TDEWallet::Backend(wallet, isPath);
KPasswordDialog *kpd = 0L;
bool emptyPass = false;
if ((isPath && TQFile::exists(wallet)) || (!isPath && TDEWallet::Backend::exists(wallet))) {
int pwless = b->open(TQByteArray());
if (0 != pwless || !b->isOpen()) {
if (pwless == 0) {
// release, start anew
delete b;
b = new TDEWallet::Backend(wallet, isPath);
}
kpd = new KPasswordDialog(KPasswordDialog::Password, false, 0);
if (appid.isEmpty()) {
kpd->setPrompt(i18n("<qt>TDE has requested to open the wallet '<b>%1</b>'. Please enter the password for this wallet below.").arg(TQStyleSheet::escape(wallet)));
} else {
kpd->setPrompt(i18n("<qt>The application '<b>%1</b>' has requested to open the wallet '<b>%2</b>'. Please enter the password for this wallet below.").arg(TQStyleSheet::escape(appid)).arg(TQStyleSheet::escape(wallet)));
}
brandNew = false;
kpd->setButtonOK(KGuiItem(i18n("&Open"),"document-open"));
} else {
emptyPass = true;
}
} else if (wallet == TDEWallet::Wallet::LocalWallet() ||
wallet == TDEWallet::Wallet::NetworkWallet()) {
// Auto create these wallets.
kpd = new KPasswordDialog(KPasswordDialog::NewPassword, false, 0);
if (appid.isEmpty()) {
kpd->setPrompt(i18n("TDE has requested to open the wallet. This is used to store sensitive data in a secure fashion. Please enter a password to use with this wallet or click cancel to deny the application's request."));
} else {
kpd->setPrompt(i18n("<qt>The application '<b>%1</b>' has requested to open the TDE wallet. This is used to store sensitive data in a secure fashion. Please enter a password to use with this wallet or click cancel to deny the application's request.").arg(TQStyleSheet::escape(appid)));
}
brandNew = true;
kpd->setButtonOK(KGuiItem(i18n("&Open"),"document-open"));
} else {
kpd = new KPasswordDialog(KPasswordDialog::NewPassword, false, 0);
if (appid.length() == 0) {
kpd->setPrompt(i18n("<qt>TDE has requested to create a new wallet named '<b>%1</b>'. Please choose a password for this wallet, or cancel to deny the application's request.").arg(TQStyleSheet::escape(wallet)));
} else {
kpd->setPrompt(i18n("<qt>The application '<b>%1</b>' has requested to create a new wallet named '<b>%2</b>'. Please choose a password for this wallet, or cancel to deny the application's request.").arg(TQStyleSheet::escape(appid)).arg(TQStyleSheet::escape(wallet)));
}
brandNew = true;
kpd->setButtonOK(KGuiItem(i18n("C&reate"),"document-new"));
}
if (kpd) {
kpd->setCaption(i18n("TDE Wallet Service"));
kpd->setAllowEmptyPasswords(true);
}
const char *p = 0L;
while (!b->isOpen()) {
assert(kpd); // kpd can't be null if isOpen() is false
setupDialog( kpd, w, appid, modal );
if (kpd->exec() == KDialog::Accepted) {
p = kpd->password();
int rc = b->open(TQByteArray().duplicate(p, strlen(p)));
if (!b->isOpen()) {
kpd->setPrompt(i18n("<qt>Error opening the wallet '<b>%1</b>'. Please try again.<br>(Error code %2: %3)").arg(TQStyleSheet::escape(wallet)).arg(rc).arg(TDEWallet::Backend::openRCToString(rc)));
kpd->clearPassword();
}
} else {
break;
}
}
if (!emptyPass && (!p || !b->isOpen())) {
delete b;
delete kpd;
return -1;
}
if (emptyPass && _openPrompt && !isAuthorizedApp(appid, wallet, w)) {
delete b;
delete kpd;
return -1;
}
_wallets.insert(rc = generateHandle(), b);
if (emptyPass) {
_passwords[wallet] = "";
} else {
_passwords[wallet] = p;
}
_handles[appid].append(rc);
delete kpd; // don't refactor this!! Argh I hate KPassDlg
if (brandNew) {
createFolder(rc, TDEWallet::Wallet::PasswordFolder());
createFolder(rc, TDEWallet::Wallet::FormDataFolder());
}
b->ref();
if (_closeIdle && _timeouts) {
_timeouts->addTimer(rc, _idleTime);
}
TQByteArray data;
TQDataStream ds(data, IO_WriteOnly);
ds << wallet;
if (brandNew) {
emitDCOPSignal("walletCreated(TQString)", data);
}
emitDCOPSignal("walletOpened(TQString)", data);
if (_wallets.count() == 1 && _launchManager) {
TDEApplication::startServiceByDesktopName("tdewalletmanager-tdewalletd");
}
} else {
if (!_handles[appid].contains(rc) && _openPrompt && !isAuthorizedApp(appid, wallet, w)) {
return -1;
}
_handles[appid].append(rc);
_wallets.find(rc)->ref();
}
return rc;
}
bool TDEWalletD::isAuthorizedApp(const TQCString& appid, const TQString& wallet, WId w) {
int response = 0;
TQCString thisApp;
if (appid.isEmpty()) {
thisApp = "TDE System";
} else {
thisApp = appid;
}
if (!implicitAllow(wallet, thisApp)) {
KBetterThanKDialogBase *dialog = new KBetterThanKDialogBase;
if (appid.isEmpty()) {
dialog->setLabel(i18n("<qt>TDE has requested access to the open wallet '<b>%1</b>'.").arg(TQStyleSheet::escape(wallet)));
} else {
dialog->setLabel(i18n("<qt>The application '<b>%1</b>' has requested access to the open wallet '<b>%2</b>'.").arg(TQStyleSheet::escape(TQString(appid))).arg(TQStyleSheet::escape(wallet)));
}
setupDialog( dialog, w, appid, false );
response = dialog->exec();
delete dialog;
}
if (response == 0 || response == 1) {
if (response == 1) {
TDEConfig cfg("tdewalletrc");
cfg.setGroup("Auto Allow");
TQStringList apps = cfg.readListEntry(wallet);
if (!apps.contains(thisApp)) {
apps += thisApp;
_implicitAllowMap[wallet] += thisApp;
cfg.writeEntry(wallet, apps);
cfg.sync();
}
}
} else if (response == 3) {
TDEConfig cfg("tdewalletrc");
cfg.setGroup("Auto Deny");
TQStringList apps = cfg.readListEntry(wallet);
if (!apps.contains(thisApp)) {
apps += thisApp;
_implicitDenyMap[wallet] += thisApp;
cfg.writeEntry(wallet, apps);
cfg.sync();
}
return false;
} else {
return false;
}
return true;
}
int TDEWalletD::deleteWallet(const TQString& wallet) {
TQString path = TDEGlobal::dirs()->saveLocation("tdewallet") + TQDir::separator() + wallet + ".kwl";
if (TQFile::exists(path)) {
close(wallet, true);
TQFile::remove(path);
TQByteArray data;
TQDataStream ds(data, IO_WriteOnly);
ds << wallet;
emitDCOPSignal("walletDeleted(TQString)", data);
return 0;
}
return -1;
}
void TDEWalletD::changePassword(const TQString& wallet, uint wId) {
TQCString appid = friendlyDCOPPeerName();
TDEWalletTransaction *xact = new TDEWalletTransaction;
xact->appid = appid;
xact->client = callingDcopClient();
xact->wallet = wallet;
xact->wId = wId;
xact->tType = TDEWalletTransaction::ChangePassword;
_transactions.append(xact);
TQTimer::singleShot(0, this, TQT_SLOT(processTransactions()));
checkActiveDialog();
}
void TDEWalletD::doTransactionChangePassword(const TQCString& appid, const TQString& wallet, uint wId) {
TQIntDictIterator<TDEWallet::Backend> it(_wallets);
TDEWallet::Backend *w = 0L;
int handle = -1;
bool reclose = false;
for (; it.current(); ++it) {
if (it.current()->walletName() == wallet) {
break;
}
}
if (!it.current()) {
handle = doTransactionOpen(appid, wallet, wId,false);
if (-1 == handle) {
KMessageBox::sorryWId(wId, i18n("Unable to open wallet. The wallet must be opened in order to change the password."), i18n("TDE Wallet Service"));
return;
}
w = _wallets.find(handle);
reclose = true;
} else {
handle = it.currentKey();
w = it.current();
}
assert(w);
KPasswordDialog *kpd;
kpd = new KPasswordDialog(KPasswordDialog::NewPassword, false, 0);
kpd->setPrompt(i18n("<qt>Please choose a new password for the wallet '<b>%1</b>'.").arg(TQStyleSheet::escape(wallet)));
kpd->setCaption(i18n("TDE Wallet Service"));
kpd->setAllowEmptyPasswords(true);
setupDialog( kpd, wId, appid, false );
if (kpd->exec() == KDialog::Accepted) {
const char *p = kpd->password();
if (p) {
_passwords[wallet] = p;
TQByteArray pa;
pa.duplicate(p, strlen(p));
int rc = w->close(pa);
if (rc < 0) {
KMessageBox::sorryWId(wId, i18n("Error re-encrypting the wallet. Password was not changed."), i18n("TDE Wallet Service"));
reclose = true;
} else {
rc = w->open(pa);
if (rc < 0) {
KMessageBox::sorryWId(wId, i18n("Error reopening the wallet. Data may be lost."), i18n("TDE Wallet Service"));
reclose = true;
}
}
}
}
delete kpd;
if (reclose) {
close(handle, true);
}
}
int TDEWalletD::close(const TQString& wallet, bool force) {
int handle = -1;
TDEWallet::Backend *w = 0L;
for (TQIntDictIterator<TDEWallet::Backend> it(_wallets);
it.current();
++it) {
if (it.current()->walletName() == wallet) {
handle = it.currentKey();
w = it.current();
break;
}
}
return closeWallet(w, handle, force);
}
int TDEWalletD::closeWallet(TDEWallet::Backend *w, int handle, bool force) {
if (w) {
const TQString& wallet = w->walletName();
assert(_passwords.contains(wallet));
if (w->refCount() == 0 || force) {
invalidateHandle(handle);
if (_closeIdle && _timeouts) {
_timeouts->removeTimer(handle);
}
_wallets.remove(handle);
if (_passwords.contains(wallet)) {
w->close(TQByteArray().duplicate(_passwords[wallet].data(), _passwords[wallet].length()));
_passwords[wallet].fill(0);
_passwords.remove(wallet);
}
doCloseSignals(handle, wallet);
delete w;
return 0;
}
return 1;
}
return -1;
}
int TDEWalletD::close(int handle, bool force) {
TQCString appid = friendlyDCOPPeerName();
TDEWallet::Backend *w = _wallets.find(handle);
bool contains = false;
if (w) { // the handle is valid
if (_handles.contains(appid)) { // we know this app
if (_handles[appid].contains(handle)) {
// the app owns this handle
_handles[appid].remove(_handles[appid].find(handle));
contains = true;
if (_handles[appid].isEmpty()) {
_handles.remove(appid);
}
}
}
// watch the side effect of the deref()
if ((contains && w->deref() == 0 && !_leaveOpen) || force) {
if (_closeIdle && _timeouts) {
_timeouts->removeTimer(handle);
}
_wallets.remove(handle);
if (force) {
invalidateHandle(handle);
}
if (_passwords.contains(w->walletName())) {
w->close(TQByteArray().duplicate(_passwords[w->walletName()].data(), _passwords[w->walletName()].length()));
_passwords[w->walletName()].fill(0);
_passwords.remove(w->walletName());
}
doCloseSignals(handle, w->walletName());
delete w;
return 0;
}
return 1; // not closed
}
return -1; // not open to begin with, or other error
}
bool TDEWalletD::isOpen(const TQString& wallet) const {
for (TQIntDictIterator<TDEWallet::Backend> it(_wallets);
it.current();
++it) {
if (it.current()->walletName() == wallet) {
return true;
}
}
return false;
}
bool TDEWalletD::isOpen(int handle) {
if (handle == 0) {
return false;
}
TDEWallet::Backend *rc = _wallets.find(handle);
if (rc == 0 && ++_failed > 5) {
_failed = 0;
TQTimer::singleShot(0, this, TQT_SLOT(notifyFailures()));
} else if (rc != 0) {
_failed = 0;
}
return rc != 0;
}
TQStringList TDEWalletD::wallets() const {
TQString path = TDEGlobal::dirs()->saveLocation("tdewallet");
TQDir dir(path, "*.kwl");
TQStringList rc;
dir.setFilter(TQDir::Files | TQDir::NoSymLinks);
const TQFileInfoList *list = dir.entryInfoList();
TQFileInfoListIterator it(*list);
TQFileInfo *fi;
while ((fi = it.current()) != 0L) {
TQString fn = fi->fileName();
if (fn.endsWith(".kwl")) {
fn.truncate(fn.length()-4);
}
rc += fn;
++it;
}
return rc;
}
void TDEWalletD::sync(int handle) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
TQByteArray p;
TQString wallet = b->walletName();
p.duplicate(_passwords[wallet].data(), _passwords[wallet].length());
b->sync(p);
p.fill(0);
}
}
TQStringList TDEWalletD::folderList(int handle) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
return b->folderList();
}
return TQStringList();
}
bool TDEWalletD::hasFolder(int handle, const TQString& f) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
return b->hasFolder(f);
}
return false;
}
bool TDEWalletD::removeFolder(int handle, const TQString& f) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
bool rc = b->removeFolder(f);
// write changes to disk immediately
TQByteArray p;
TQString wallet = b->walletName();
p.duplicate(_passwords[wallet].data(), _passwords[wallet].length());
b->sync(p);
p.fill(0);
TQByteArray data;
TQDataStream ds(data, IO_WriteOnly);
ds << b->walletName();
emitDCOPSignal("folderListUpdated(TQString)", data);
return rc;
}
return false;
}
bool TDEWalletD::createFolder(int handle, const TQString& f) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
bool rc = b->createFolder(f);
// write changes to disk immediately
TQByteArray p;
TQString wallet = b->walletName();
p.duplicate(_passwords[wallet].data(), _passwords[wallet].length());
b->sync(p);
p.fill(0);
TQByteArray data;
TQDataStream ds(data, IO_WriteOnly);
ds << b->walletName();
emitDCOPSignal("folderListUpdated(TQString)", data);
return rc;
}
return false;
}
TQByteArray TDEWalletD::readMap(int handle, const TQString& folder, const TQString& key) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
b->setFolder(folder);
TDEWallet::Entry *e = b->readEntry(key);
if (e && e->type() == TDEWallet::Wallet::Map) {
return e->map();
}
}
return TQByteArray();
}
TQMap<TQString,TQByteArray> TDEWalletD::readMapList(int handle, const TQString& folder, const TQString& key) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
b->setFolder(folder);
TQPtrList<TDEWallet::Entry> e = b->readEntryList(key);
TQMap<TQString, TQByteArray> rc;
TQPtrListIterator<TDEWallet::Entry> it(e);
TDEWallet::Entry *entry;
while ((entry = it.current())) {
if (entry->type() == TDEWallet::Wallet::Map) {
rc.insert(entry->key(), entry->map());
}
++it;
}
return rc;
}
return TQMap<TQString, TQByteArray>();
}
TQByteArray TDEWalletD::readEntry(int handle, const TQString& folder, const TQString& key) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
b->setFolder(folder);
TDEWallet::Entry *e = b->readEntry(key);
if (e) {
return e->value();
}
}
return TQByteArray();
}
TQMap<TQString, TQByteArray> TDEWalletD::readEntryList(int handle, const TQString& folder, const TQString& key) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
b->setFolder(folder);
TQPtrList<TDEWallet::Entry> e = b->readEntryList(key);
TQMap<TQString, TQByteArray> rc;
TQPtrListIterator<TDEWallet::Entry> it(e);
TDEWallet::Entry *entry;
while ((entry = it.current())) {
rc.insert(entry->key(), entry->value());
++it;
}
return rc;
}
return TQMap<TQString, TQByteArray>();
}
TQStringList TDEWalletD::entryList(int handle, const TQString& folder) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
b->setFolder(folder);
return b->entryList();
}
return TQStringList();
}
TQString TDEWalletD::readPassword(int handle, const TQString& folder, const TQString& key) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
b->setFolder(folder);
TDEWallet::Entry *e = b->readEntry(key);
if (e && e->type() == TDEWallet::Wallet::Password) {
return e->password();
}
}
return TQString::null;
}
TQMap<TQString, TQString> TDEWalletD::readPasswordList(int handle, const TQString& folder, const TQString& key) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
b->setFolder(folder);
TQPtrList<TDEWallet::Entry> e = b->readEntryList(key);
TQMap<TQString, TQString> rc;
TQPtrListIterator<TDEWallet::Entry> it(e);
TDEWallet::Entry *entry;
while ((entry = it.current())) {
if (entry->type() == TDEWallet::Wallet::Password) {
rc.insert(entry->key(), entry->password());
}
++it;
}
return rc;
}
return TQMap<TQString, TQString>();
}
int TDEWalletD::writeMap(int handle, const TQString& folder, const TQString& key, const TQByteArray& value) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
b->setFolder(folder);
TDEWallet::Entry e;
e.setKey(key);
e.setValue(value);
e.setType(TDEWallet::Wallet::Map);
b->writeEntry(&e);
// write changes to disk immediately
TQByteArray p;
TQString wallet = b->walletName();
p.duplicate(_passwords[wallet].data(), _passwords[wallet].length());
b->sync(p);
p.fill(0);
emitFolderUpdated(b->walletName(), folder);
return 0;
}
return -1;
}
int TDEWalletD::writeEntry(int handle, const TQString& folder, const TQString& key, const TQByteArray& value, int entryType) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
b->setFolder(folder);
TDEWallet::Entry e;
e.setKey(key);
e.setValue(value);
e.setType(TDEWallet::Wallet::EntryType(entryType));
b->writeEntry(&e);
// write changes to disk immediately
TQByteArray p;
TQString wallet = b->walletName();
p.duplicate(_passwords[wallet].data(), _passwords[wallet].length());
b->sync(p);
p.fill(0);
emitFolderUpdated(b->walletName(), folder);
return 0;
}
return -1;
}
int TDEWalletD::writeEntry(int handle, const TQString& folder, const TQString& key, const TQByteArray& value) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
b->setFolder(folder);
TDEWallet::Entry e;
e.setKey(key);
e.setValue(value);
e.setType(TDEWallet::Wallet::Stream);
b->writeEntry(&e);
// write changes to disk immediately
TQByteArray p;
TQString wallet = b->walletName();
p.duplicate(_passwords[wallet].data(), _passwords[wallet].length());
b->sync(p);
p.fill(0);
emitFolderUpdated(b->walletName(), folder);
return 0;
}
return -1;
}
int TDEWalletD::writePassword(int handle, const TQString& folder, const TQString& key, const TQString& value) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
b->setFolder(folder);
TDEWallet::Entry e;
e.setKey(key);
e.setValue(value);
e.setType(TDEWallet::Wallet::Password);
b->writeEntry(&e);
// write changes to disk immediately
TQByteArray p;
TQString wallet = b->walletName();
p.duplicate(_passwords[wallet].data(), _passwords[wallet].length());
b->sync(p);
p.fill(0);
emitFolderUpdated(b->walletName(), folder);
return 0;
}
return -1;
}
int TDEWalletD::entryType(int handle, const TQString& folder, const TQString& key) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
if (!b->hasFolder(folder)) {
return TDEWallet::Wallet::Unknown;
}
b->setFolder(folder);
if (b->hasEntry(key)) {
return b->readEntry(key)->type();
}
}
return TDEWallet::Wallet::Unknown;
}
bool TDEWalletD::hasEntry(int handle, const TQString& folder, const TQString& key) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
if (!b->hasFolder(folder)) {
return false;
}
b->setFolder(folder);
return b->hasEntry(key);
}
return false;
}
int TDEWalletD::removeEntry(int handle, const TQString& folder, const TQString& key) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
if (!b->hasFolder(folder)) {
return 0;
}
b->setFolder(folder);
bool rc = b->removeEntry(key);
// write changes to disk immediately
TQByteArray p;
TQString wallet = b->walletName();
p.duplicate(_passwords[wallet].data(), _passwords[wallet].length());
b->sync(p);
p.fill(0);
emitFolderUpdated(b->walletName(), folder);
return rc ? 0 : -3;
}
return -1;
}
void TDEWalletD::slotAppUnregistered(const TQCString& app) {
if (_handles.contains(app)) {
TQValueList<int> l = _handles[app];
for (TQValueList<int>::Iterator i = l.begin(); i != l.end(); ++i) {
_handles[app].remove(*i);
TDEWallet::Backend *w = _wallets.find(*i);
if (w && !_leaveOpen && 0 == w->deref()) {
close(w->walletName(), true);
}
}
_handles.remove(app);
}
}
void TDEWalletD::invalidateHandle(int handle) {
for (TQMap<TQCString,TQValueList<int> >::Iterator i = _handles.begin();
i != _handles.end();
++i) {
i.data().remove(handle);
}
}
TDEWallet::Backend *TDEWalletD::getWallet(const TQCString& appid, int handle) {
if (handle == 0) {
return 0L;
}
TDEWallet::Backend *w = _wallets.find(handle);
if (w) { // the handle is valid
if (_handles.contains(appid)) { // we know this app
if (_handles[appid].contains(handle)) {
// the app owns this handle
_failed = 0;
if (_closeIdle && _timeouts) {
_timeouts->resetTimer(handle, _idleTime);
}
return w;
}
}
}
if (++_failed > 5) {
_failed = 0;
TQTimer::singleShot(0, this, TQT_SLOT(notifyFailures()));
}
return 0L;
}
void TDEWalletD::notifyFailures() {
if (!_showingFailureNotify) {
_showingFailureNotify = true;
KMessageBox::information(0, i18n("There have been repeated failed attempts to gain access to a wallet. An application may be misbehaving."), i18n("TDE Wallet Service"));
_showingFailureNotify = false;
}
}
void TDEWalletD::doCloseSignals(int handle, const TQString& wallet) {
TQByteArray data;
TQDataStream ds(data, IO_WriteOnly);
ds << handle;
emitDCOPSignal("walletClosed(int)", data);
TQByteArray data2;
TQDataStream ds2(data2, IO_WriteOnly);
ds2 << wallet;
emitDCOPSignal("walletClosed(TQString)", data2);
if (_wallets.isEmpty()) {
emitDCOPSignal("allWalletsClosed()", TQByteArray());
}
}
int TDEWalletD::renameEntry(int handle, const TQString& folder, const TQString& oldName, const TQString& newName) {
TDEWallet::Backend *b;
if ((b = getWallet(friendlyDCOPPeerName(), handle))) {
b->setFolder(folder);
int rc = b->renameEntry(oldName, newName);
// write changes to disk immediately
TQByteArray p;
TQString wallet = b->walletName();
p.duplicate(_passwords[wallet].data(), _passwords[wallet].length());
b->sync(p);
p.fill(0);
emitFolderUpdated(b->walletName(), folder);
return rc;
}
return -1;
}
TQStringList TDEWalletD::users(const TQString& wallet) const {
TQStringList rc;
for (TQIntDictIterator<TDEWallet::Backend> it(_wallets);
it.current();
++it) {
if (it.current()->walletName() == wallet) {
for (TQMap<TQCString,TQValueList<int> >::ConstIterator hit = _handles.begin(); hit != _handles.end(); ++hit) {
if (hit.data().contains(it.currentKey())) {
rc += hit.key();
}
}
break;
}
}
return rc;
}
bool TDEWalletD::disconnectApplication(const TQString& wallet, const TQCString& application) {
for (TQIntDictIterator<TDEWallet::Backend> it(_wallets);
it.current();
++it) {
if (it.current()->walletName() == wallet) {
if (_handles[application].contains(it.currentKey())) {
_handles[application].remove(it.currentKey());
if (_handles[application].isEmpty()) {
_handles.remove(application);
}
if (it.current()->deref() == 0) {
close(it.current()->walletName(), true);
}
TQByteArray data;
TQDataStream ds(data, IO_WriteOnly);
ds << wallet;
ds << application;
emitDCOPSignal("applicationDisconnected(TQString,TQCString)", data);
return true;
}
}
}
return false;
}
void TDEWalletD::emitFolderUpdated(const TQString& wallet, const TQString& folder) {
TQByteArray data;
TQDataStream ds(data, IO_WriteOnly);
ds << wallet;
ds << folder;
emitDCOPSignal("folderUpdated(TQString,TQString)", data);
}
void TDEWalletD::emitWalletListDirty() {
emitDCOPSignal("walletListDirty()", TQByteArray());
}
void TDEWalletD::reconfigure() {
TDEConfig cfg("tdewalletrc");
cfg.setGroup("Wallet");
_firstUse = cfg.readBoolEntry("First Use", true);
_enabled = cfg.readBoolEntry("Enabled", true);
_launchManager = cfg.readBoolEntry("Launch Manager", true);
_leaveOpen = cfg.readBoolEntry("Leave Open", false);
bool idleSave = _closeIdle;
_closeIdle = cfg.readBoolEntry("Close When Idle", false);
_openPrompt = cfg.readBoolEntry("Prompt on Open", true);
int timeSave = _idleTime;
// in minutes!
_idleTime = cfg.readNumEntry("Idle Timeout", 10) * 60 * 1000;
if (cfg.readBoolEntry("Close on Screensaver", false)) {
connectDCOPSignal("kdesktop", "KScreensaverIface", "KDE_start_screensaver()", "closeAllWallets()", false);
} else {
disconnectDCOPSignal("kdesktop", "KScreensaverIface", "KDE_start_screensaver()", "closeAllWallets()");
}
// Handle idle changes
if (_closeIdle) {
if (_idleTime != timeSave) { // Timer length changed
TQIntDictIterator<TDEWallet::Backend> it(_wallets);
for (; it.current(); ++it) {
_timeouts->resetTimer(it.currentKey(), _idleTime);
}
}
if (!idleSave) { // add timers for all the wallets
TQIntDictIterator<TDEWallet::Backend> it(_wallets);
for (; it.current(); ++it) {
_timeouts->addTimer(it.currentKey(), _idleTime);
}
}
} else {
_timeouts->clear();
}
// Update the implicit allow stuff
_implicitAllowMap.clear();
cfg.setGroup("Auto Allow");
TQStringList entries = cfg.entryMap("Auto Allow").keys();
for (TQStringList::Iterator i = entries.begin(); i != entries.end(); ++i) {
_implicitAllowMap[*i] = cfg.readListEntry(*i);
}
// Update the implicit allow stuff
_implicitDenyMap.clear();
cfg.setGroup("Auto Deny");
entries = cfg.entryMap("Auto Deny").keys();
for (TQStringList::Iterator i = entries.begin(); i != entries.end(); ++i) {
_implicitDenyMap[*i] = cfg.readListEntry(*i);
}
// Update if wallet was enabled/disabled
if (!_enabled) { // close all wallets
while (!_wallets.isEmpty()) {
TQIntDictIterator<TDEWallet::Backend> it(_wallets);
if (!it.current()) { // necessary?
break;
}
closeWallet(it.current(), it.currentKey(), true);
}
}
}
bool TDEWalletD::isEnabled() const {
return _enabled;
}
bool TDEWalletD::folderDoesNotExist(const TQString& wallet, const TQString& folder) {
if (!wallets().contains(wallet)) {
return true;
}
for (TQIntDictIterator<TDEWallet::Backend> it(_wallets); it.current(); ++it) {
if (it.current()->walletName() == wallet) {
return it.current()->folderDoesNotExist(folder);
}
}
TDEWallet::Backend *b = new TDEWallet::Backend(wallet);
b->open(TQByteArray());
bool rc = b->folderDoesNotExist(folder);
delete b;
return rc;
}
bool TDEWalletD::keyDoesNotExist(const TQString& wallet, const TQString& folder, const TQString& key) {
if (!wallets().contains(wallet)) {
return true;
}
for (TQIntDictIterator<TDEWallet::Backend> it(_wallets); it.current(); ++it) {
if (it.current()->walletName() == wallet) {
return it.current()->entryDoesNotExist(folder, key);
}
}
TDEWallet::Backend *b = new TDEWallet::Backend(wallet);
b->open(TQByteArray());
bool rc = b->entryDoesNotExist(folder, key);
delete b;
return rc;
}
bool TDEWalletD::implicitAllow(const TQString& wallet, const TQCString& app) {
return _implicitAllowMap[wallet].contains(TQString::fromLocal8Bit(app));
}
bool TDEWalletD::implicitDeny(const TQString& wallet, const TQCString& app) {
return _implicitDenyMap[wallet].contains(TQString::fromLocal8Bit(app));
}
TQCString TDEWalletD::friendlyDCOPPeerName() {
DCOPClient *dc = callingDcopClient();
if (!dc) {
return "";
}
return dc->senderId().replace(TQRegExp("-[0-9]+$"), "");
}
void TDEWalletD::timedOut(int id) {
TDEWallet::Backend *w = _wallets.find(id);
if (w) {
closeWallet(w, id, true);
}
}
void TDEWalletD::closeAllWallets() {
TQIntDict<TDEWallet::Backend> tw = _wallets;
for (TQIntDictIterator<TDEWallet::Backend> it(tw); it.current(); ++it) {
closeWallet(it.current(), it.currentKey(), true);
}
tw.clear();
// All of this should be basically noop. Let's just be safe.
_wallets.clear();
for (TQMap<TQString,TQCString>::Iterator it = _passwords.begin();
it != _passwords.end();
++it) {
it.data().fill(0);
}
_passwords.clear();
}
TQString TDEWalletD::networkWallet() {
return TDEWallet::Wallet::NetworkWallet();
}
TQString TDEWalletD::localWallet() {
return TDEWallet::Wallet::LocalWallet();
}
#include "tdewalletd.moc"