mirror of
https://github.com/nextcloud/desktop.git
synced 2024-12-21 05:13:04 +03:00
281 lines
9.1 KiB
C++
281 lines
9.1 KiB
C++
/*
|
|
* Copyright (C) by Olivier Goffart <ogoffart@woboq.com>
|
|
*
|
|
* 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; version 2 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.
|
|
*/
|
|
|
|
#include "accountmanager.h"
|
|
#include "sslerrordialog.h"
|
|
#include "proxyauthhandler.h"
|
|
#include <theme.h>
|
|
#include <creds/credentialsfactory.h>
|
|
#include <creds/abstractcredentials.h>
|
|
#include <cookiejar.h>
|
|
#include <QSettings>
|
|
#include <QDir>
|
|
#include <QNetworkAccessManager>
|
|
|
|
namespace {
|
|
static const char urlC[] = "url";
|
|
static const char authTypeC[] = "authType";
|
|
static const char userC[] = "user";
|
|
static const char httpUserC[] = "http_user";
|
|
static const char caCertsKeyC[] = "CaCertificates";
|
|
static const char accountsC[] = "Accounts";
|
|
}
|
|
|
|
|
|
namespace OCC {
|
|
|
|
AccountManager *AccountManager::instance()
|
|
{
|
|
static AccountManager instance;
|
|
return &instance;
|
|
}
|
|
|
|
bool AccountManager::restore()
|
|
{
|
|
auto settings = Account::settingsWithGroup(QLatin1String(accountsC));
|
|
|
|
// If there are no accounts, check the old format.
|
|
if (settings->childGroups().isEmpty()) {
|
|
return restoreFromLegacySettings();
|
|
}
|
|
|
|
foreach (const auto& accountId, settings->childGroups()) {
|
|
settings->beginGroup(accountId);
|
|
if (auto acc = load(*settings)) {
|
|
acc->_id = accountId;
|
|
addAccount(acc);
|
|
}
|
|
settings->endGroup();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool AccountManager::restoreFromLegacySettings()
|
|
{
|
|
// try to open the correctly themed settings
|
|
auto settings = Account::settingsWithGroup(Theme::instance()->appName());
|
|
|
|
bool migratedCreds = false;
|
|
|
|
// if the settings file could not be opened, the childKeys list is empty
|
|
// then try to load settings from a very old place
|
|
if( settings->childKeys().isEmpty() ) {
|
|
// Now try to open the original ownCloud settings to see if they exist.
|
|
QString oCCfgFile = QDir::fromNativeSeparators( settings->fileName() );
|
|
// replace the last two segments with ownCloud/owncloud.cfg
|
|
oCCfgFile = oCCfgFile.left( oCCfgFile.lastIndexOf('/'));
|
|
oCCfgFile = oCCfgFile.left( oCCfgFile.lastIndexOf('/'));
|
|
oCCfgFile += QLatin1String("/ownCloud/owncloud.cfg");
|
|
|
|
qDebug() << "Migrate: checking old config " << oCCfgFile;
|
|
|
|
QFileInfo fi( oCCfgFile );
|
|
if( fi.isReadable() ) {
|
|
QSettings *oCSettings = new QSettings(oCCfgFile, QSettings::IniFormat);
|
|
oCSettings->beginGroup(QLatin1String("ownCloud"));
|
|
|
|
// Check the theme url to see if it is the same url that the oC config was for
|
|
QString overrideUrl = Theme::instance()->overrideServerUrl();
|
|
if( !overrideUrl.isEmpty() ) {
|
|
if (overrideUrl.endsWith('/')) { overrideUrl.chop(1); }
|
|
QString oCUrl = oCSettings->value(QLatin1String(urlC)).toString();
|
|
if (oCUrl.endsWith('/')) { oCUrl.chop(1); }
|
|
|
|
// in case the urls are equal reset the settings object to read from
|
|
// the ownCloud settings object
|
|
qDebug() << "Migrate oC config if " << oCUrl << " == " << overrideUrl << ":"
|
|
<< (oCUrl == overrideUrl ? "Yes" : "No");
|
|
if( oCUrl == overrideUrl ) {
|
|
migratedCreds = true;
|
|
settings.reset( oCSettings );
|
|
} else {
|
|
delete oCSettings;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Try to load the single account.
|
|
if (!settings->childKeys().isEmpty()) {
|
|
if (auto acc = load(*settings)) {
|
|
if (migratedCreds) {
|
|
acc->setMigrated(true);
|
|
}
|
|
addAccount(acc);
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void AccountManager::save(bool saveCredentials)
|
|
{
|
|
auto settings = Account::settingsWithGroup(QLatin1String(accountsC));
|
|
foreach (const auto &acc, _accounts) {
|
|
settings->beginGroup(acc->account()->id());
|
|
save(acc->account(), *settings, saveCredentials);
|
|
settings->endGroup();
|
|
}
|
|
}
|
|
|
|
void AccountManager::wantsAccountSavedSlot(AccountPtr a)
|
|
{
|
|
qDebug() << "Saving account" << a->url().toString();
|
|
auto settings = Account::settingsWithGroup(QLatin1String(accountsC));
|
|
settings->beginGroup(a->id());
|
|
save(a, *settings, false); // don't save credentials they might not have been loaded yet
|
|
settings->endGroup();
|
|
}
|
|
|
|
void AccountManager::save(const AccountPtr& acc, QSettings& settings, bool saveCredentials)
|
|
{
|
|
settings.setValue(QLatin1String(urlC), acc->_url.toString());
|
|
if (acc->_credentials) {
|
|
if (saveCredentials) {
|
|
// Only persist the credentials if the parameter is set, on migration from 1.8.x
|
|
// we want to save the accounts but not overwrite the credentials
|
|
// (This is easier than asynchronously fetching the credentials from keychain and then
|
|
// re-persisting them)
|
|
acc->_credentials->persist();
|
|
}
|
|
Q_FOREACH(QString key, acc->_settingsMap.keys()) {
|
|
settings.setValue(key, acc->_settingsMap.value(key));
|
|
}
|
|
settings.setValue(QLatin1String(authTypeC), acc->_credentials->authType());
|
|
|
|
// HACK: Save http_user also as user
|
|
if (acc->_settingsMap.contains(httpUserC))
|
|
settings.setValue(userC, acc->_settingsMap.value(httpUserC));
|
|
}
|
|
settings.sync();
|
|
|
|
// Save accepted certificates.
|
|
settings.beginGroup(QLatin1String("General"));
|
|
qDebug() << "Saving " << acc->approvedCerts().count() << " unknown certs.";
|
|
QByteArray certs;
|
|
Q_FOREACH( const QSslCertificate& cert, acc->approvedCerts() ) {
|
|
certs += cert.toPem() + '\n';
|
|
}
|
|
if (!certs.isEmpty()) {
|
|
settings.setValue( QLatin1String(caCertsKeyC), certs );
|
|
}
|
|
settings.endGroup();
|
|
|
|
// Save cookies.
|
|
if (acc->_am) {
|
|
CookieJar* jar = qobject_cast<CookieJar*>(acc->_am->cookieJar());
|
|
if (jar) {
|
|
qDebug() << "Saving cookies.";
|
|
jar->save();
|
|
}
|
|
}
|
|
}
|
|
|
|
AccountPtr AccountManager::load(QSettings& settings)
|
|
{
|
|
auto acc = createAccount();
|
|
|
|
acc->setUrl(settings.value(QLatin1String(urlC)).toUrl());
|
|
|
|
// We want to only restore settings for that auth type and the user value
|
|
acc->_settingsMap.insert(QLatin1String(userC), settings.value(userC));
|
|
QString authTypePrefix = settings.value(authTypeC).toString() + "_";
|
|
Q_FOREACH(QString key, settings.childKeys()) {
|
|
if (!key.startsWith(authTypePrefix))
|
|
continue;
|
|
acc->_settingsMap.insert(key, settings.value(key));
|
|
}
|
|
|
|
acc->setCredentials(CredentialsFactory::create(settings.value(QLatin1String(authTypeC)).toString()));
|
|
|
|
// now the cert, it is in the general group
|
|
settings.beginGroup(QLatin1String("General"));
|
|
acc->setApprovedCerts(QSslCertificate::fromData(settings.value(caCertsKeyC).toByteArray()));
|
|
settings.endGroup();
|
|
|
|
return acc;
|
|
}
|
|
|
|
AccountState *AccountManager::addAccount(const AccountPtr& newAccount)
|
|
{
|
|
auto id = newAccount->id();
|
|
if (id.isEmpty() || !isAccountIdAvailable(id)) {
|
|
id = generateFreeAccountId();
|
|
}
|
|
newAccount->_id = id;
|
|
|
|
QObject::connect(newAccount.data(), SIGNAL(wantsAccountSaved(AccountPtr)),
|
|
this, SLOT(wantsAccountSavedSlot(AccountPtr)));
|
|
|
|
AccountStatePtr newAccountState(new AccountState(newAccount));
|
|
_accounts << newAccountState;
|
|
emit accountAdded(newAccountState.data());
|
|
return newAccountState.data();
|
|
}
|
|
|
|
void AccountManager::deleteAccount(AccountState* account)
|
|
{
|
|
auto it = std::find(_accounts.begin(), _accounts.end(), account);
|
|
if (it == _accounts.end()) { return; }
|
|
auto copy = *it; // keep a reference to the shared pointer so it does not delete it just yet
|
|
_accounts.erase(it);
|
|
|
|
auto settings = Account::settingsWithGroup(QLatin1String(accountsC));
|
|
settings->remove(account->account()->id());
|
|
|
|
emit accountRemoved(account);
|
|
}
|
|
|
|
AccountPtr AccountManager::createAccount()
|
|
{
|
|
AccountPtr acc = Account::create();
|
|
acc->setSslErrorHandler(new SslDialogErrorHandler);
|
|
connect(acc.data(), SIGNAL(proxyAuthenticationRequired(QNetworkProxy, QAuthenticator*)),
|
|
ProxyAuthHandler::instance(), SLOT(handleProxyAuthenticationRequired(QNetworkProxy,QAuthenticator*)));
|
|
return acc;
|
|
}
|
|
|
|
|
|
void AccountManager::shutdown()
|
|
{
|
|
auto accountsCopy = _accounts;
|
|
_accounts.clear();
|
|
foreach (const auto &acc, accountsCopy) {
|
|
emit accountRemoved(acc.data());
|
|
}
|
|
}
|
|
|
|
bool AccountManager::isAccountIdAvailable(const QString& id) const
|
|
{
|
|
foreach (const auto& acc, _accounts) {
|
|
if (acc->account()->id() == id) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
QString AccountManager::generateFreeAccountId() const
|
|
{
|
|
int i = 0;
|
|
forever {
|
|
QString id = QString::number(i);
|
|
if (isAccountIdAvailable(id)) {
|
|
return id;
|
|
}
|
|
++i;
|
|
}
|
|
}
|
|
|
|
}
|