diff --git a/kcms/users/src/user.cpp b/kcms/users/src/user.cpp index 74ecdf298..7699ce980 100644 --- a/kcms/users/src/user.cpp +++ b/kcms/users/src/user.cpp @@ -1,261 +1,262 @@ /* Copyright 2019 Nicolas Fella Copyright 2020 Carson Black This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) version 3, or any later version accepted by the membership of KDE e.V. (or its successor approved by the membership of KDE e.V.), which shall act as a proxy defined in Section 6 of version 3 of the license. 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library. If not, see . */ -#include +#include "user.h" +#include "user_interface.h" + #include #include #include #include -#include "user_interface.h" User::User(QObject* parent) : QObject(parent) {} int User::uid() const { return mUid; } void User::setUid(int value) { if (mUid == value) { return; } mUid = value; Q_EMIT uidChanged(value); } QString User::name() const { return mName; } void User::setName(const QString &value) { if (mName == value) { return; } mName = value; Q_EMIT nameChanged(value); } QString User::realName() const { return mRealName; } void User::setRealName(const QString &value) { if (mRealName == value) { return; } mRealName = value; Q_EMIT realNameChanged(value); } QString User::email() const { return mEmail; } void User::setEmail(const QString &value) { if (mEmail == value) { return; } mEmail = value; Q_EMIT emailChanged(value); } QUrl User::face() const { return mFace; } bool User::faceValid() const { return mFaceValid; } void User::setFace(const QUrl &value) { if (mFace == value) { return; } mFace = value; mFaceValid = QFile::exists(value.path()); Q_EMIT faceValidChanged(mFaceValid); Q_EMIT faceChanged(value); } bool User::administrator() const { return mAdministrator; } void User::setAdministrator(bool value) { if (mAdministrator == value) { return; } mAdministrator = value; Q_EMIT administratorChanged(value); } void User::setPath(const QDBusObjectPath &path) { if (!m_dbusIface.isNull()) delete m_dbusIface; m_dbusIface = new OrgFreedesktopAccountsUserInterface(QStringLiteral("org.freedesktop.Accounts"), path.path(), QDBusConnection::systemBus(), this); if (!m_dbusIface->isValid() || m_dbusIface->lastError().isValid() || m_dbusIface->systemAccount()) { return; } mPath = path; auto update = [=]() { bool userDataChanged = false; if (mUid != m_dbusIface->uid()) { mUid = m_dbusIface->uid(); userDataChanged = true; Q_EMIT uidChanged(mUid); } if (mName != m_dbusIface->userName()) { mName = m_dbusIface->userName(); userDataChanged = true; Q_EMIT nameChanged(mName); } if (mFace != QUrl(m_dbusIface->iconFile())) { mFace = QUrl(m_dbusIface->iconFile()); mFaceValid = QFileInfo::exists(mFace.toString()); userDataChanged = true; Q_EMIT faceChanged(mFace); Q_EMIT faceValidChanged(mFaceValid); } if (mRealName != m_dbusIface->realName()) { mRealName = m_dbusIface->realName(); userDataChanged = true; Q_EMIT realNameChanged(mRealName); } if (mEmail != m_dbusIface->email()) { mEmail = m_dbusIface->email(); userDataChanged = true; Q_EMIT emailChanged(mEmail); } const auto administrator = (m_dbusIface->accountType() == 1); if (mAdministrator != administrator) { mAdministrator = administrator; userDataChanged = true; Q_EMIT administratorChanged(mAdministrator); } const auto loggedIn = (mUid == getuid()); if (mLoggedIn != loggedIn) { mLoggedIn = loggedIn; userDataChanged = true; } if (userDataChanged) { Q_EMIT dataChanged(); } }; connect(m_dbusIface, &OrgFreedesktopAccountsUserInterface::Changed, [=]() { update(); }); update(); } static char saltCharacter() { static constexpr const quint32 letterCount = 64; static const char saltCharacters[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZ" "abcdefghijklmnopqrstuvwxyz" "./0123456789"; // and trailing NUL static_assert(sizeof(saltCharacters) == (letterCount+1), // 64 letters and trailing NUL "Salt-chars array is not exactly 64 letters long"); const quint32 index = QRandomGenerator::system()->bounded(0u, letterCount); return saltCharacters[index]; } static QString saltPassword(const QString &plain) { QString salt; salt.append("$6$"); for (auto i = 0; i < 16; i++) { salt.append(saltCharacter()); } salt.append("$"); auto stdStrPlain = plain.toStdString(); auto cStrPlain = stdStrPlain.c_str(); auto stdStrSalt = salt.toStdString(); auto cStrSalt = stdStrSalt.c_str(); auto salted = crypt(cStrPlain, cStrSalt); return QString::fromUtf8(salted); } void User::setPassword(const QString &password) { m_dbusIface->SetPassword(saltPassword(password), QString()); } QDBusObjectPath User::path() const { return mPath; } void User::apply() { auto job = new UserApplyJob(m_dbusIface, mName, mEmail, mRealName, mFace.toString().replace("file://", ""), mAdministrator ? 1 : 0); job->start(); } bool User::loggedIn() const { return mLoggedIn; } UserApplyJob::UserApplyJob(QPointer dbusIface, QString name, QString email, QString realname, QString icon, int type) : KJob(), m_name(name), m_email(email), m_realname(realname), m_icon(icon), m_type(type), m_dbusIface(dbusIface) { } void UserApplyJob::start() { const std::map (OrgFreedesktopAccountsUserInterface::*)(const QString&)> set = { {m_name, &OrgFreedesktopAccountsUserInterface::SetUserName}, {m_email, &OrgFreedesktopAccountsUserInterface::SetEmail}, {m_realname, &OrgFreedesktopAccountsUserInterface::SetRealName}, {m_icon, &OrgFreedesktopAccountsUserInterface::SetIconFile} }; // Do our dbus invocations with blocking calls, since the accounts service // will return permission denied if there's a polkit dialog open while a // request is made. for (auto const &x: set) { (m_dbusIface->*(x.second))(x.first).waitForFinished(); } m_dbusIface->SetAccountType(m_type).waitForFinished(); emitResult(); }