Some issue with UserProfile

This commit is contained in:
CH Chethan Reddy 2020-07-01 17:47:10 +05:30
parent 6fae36abc4
commit ac1fbbb69f
10 changed files with 154 additions and 115 deletions

View file

@ -15,6 +15,9 @@ set(CMAKE_CXX_STANDARD 17 CACHE STRING "C++ standard")
set(CMAKE_CXX_STANDARD_REQUIRED ON CACHE BOOL "Require C++ standard to be supported") set(CMAKE_CXX_STANDARD_REQUIRED ON CACHE BOOL "Require C++ standard to be supported")
set(CMAKE_POSITION_INDEPENDENT_CODE ON CACHE BOOL "compile as PIC by default") set(CMAKE_POSITION_INDEPENDENT_CODE ON CACHE BOOL "compile as PIC by default")
# set (CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -fno-omit-frame-pointer -fsanitize=address")
# set (CMAKE_LINKER_FLAGS_DEBUG "${CMAKE_LINKER_FLAGS_DEBUG} -fno-omit-frame-pointer -fsanitize=address")
option(HUNTER_ENABLED "Enable Hunter package manager" OFF) option(HUNTER_ENABLED "Enable Hunter package manager" OFF)
include("cmake/HunterGate.cmake") include("cmake/HunterGate.cmake")
HunterGate( HunterGate(

View file

@ -19,17 +19,6 @@ ApplicationWindow{
Layout.alignment: Qt.AlignHCenter Layout.alignment: Qt.AlignHCenter
palette: colors palette: colors
UserProfileList{
id: userProfileList
userId: user_data.userId
onUserIdChanged : {
userProfileList.updateDeviceList()
}
onDeviceListUpdated : {
modelDeviceList.deviceList = userProfileList
}
}
Component { Component {
id: deviceVerificationDialog id: deviceVerificationDialog
DeviceVerification {} DeviceVerification {}
@ -94,7 +83,7 @@ ApplicationWindow{
ToolTip.visible: hovered ToolTip.visible: hovered
ToolTip.text: qsTr("Ban the user") ToolTip.text: qsTr("Ban the user")
onClicked : { onClicked : {
userProfileList.banUser() modelDeviceList.deviceList.banUser()
} }
} }
// ImageButton{ // ImageButton{
@ -106,7 +95,7 @@ ApplicationWindow{
// ToolTip.visible: hovered // ToolTip.visible: hovered
// ToolTip.text: qsTr("Ignore messages from this user") // ToolTip.text: qsTr("Ignore messages from this user")
// onClicked : { // onClicked : {
// userProfileList.ignoreUser() // modelDeviceList.deviceList.ignoreUser()
// } // }
// } // }
ImageButton{ ImageButton{
@ -118,7 +107,7 @@ ApplicationWindow{
ToolTip.visible: hovered ToolTip.visible: hovered
ToolTip.text: qsTr("Start a private chat") ToolTip.text: qsTr("Start a private chat")
onClicked : { onClicked : {
userProfileList.startChat() modelDeviceList.deviceList.startChat()
} }
} }
ImageButton{ ImageButton{
@ -130,7 +119,7 @@ ApplicationWindow{
ToolTip.visible: hovered ToolTip.visible: hovered
ToolTip.text: qsTr("Kick the user") ToolTip.text: qsTr("Kick the user")
onClicked : { onClicked : {
userProfileList.kickUser() modelDeviceList.deviceList.kickUser()
} }
} }
} }
@ -142,13 +131,14 @@ ApplicationWindow{
Layout.alignment: Qt.AlignHCenter Layout.alignment: Qt.AlignHCenter
ListView{ ListView{
id: deviceList id: devicelist
anchors.fill: parent anchors.fill: parent
clip: true clip: true
spacing: 4 spacing: 4
model: UserProfileModel{ model: UserProfileModel{
id: modelDeviceList id: modelDeviceList
deviceList.userId : user_data.userId
} }
delegate: RowLayout{ delegate: RowLayout{
@ -157,13 +147,21 @@ ApplicationWindow{
top : 50 top : 50
} }
ColumnLayout{ ColumnLayout{
RowLayout{
Text{ Text{
Layout.fillWidth: true Layout.fillWidth: true
color: colors.text color: colors.text
font.bold: true font.bold: true
Layout.alignment: Qt.AlignRight Layout.alignment: Qt.AlignLeft
text: deviceID text: deviceID
} }
Text{
Layout.fillWidth: true
color:colors.text
Layout.alignment: Qt.AlignLeft
text: (verified_status == UserProfileList.VERIFIED?"V":(verified_status == UserProfileList.UNVERIFIED?"NV":"B"))
}
}
Text{ Text{
Layout.fillWidth: true Layout.fillWidth: true
color:colors.text color:colors.text

View file

@ -2885,7 +2885,6 @@ Cache::statusMessage(const std::string &user_id)
void void
to_json(json &j, const UserCache &info) to_json(json &j, const UserCache &info)
{ {
j["user_id"] = info.user_id;
j["is_user_verified"] = info.is_user_verified; j["is_user_verified"] = info.is_user_verified;
j["cross_verified"] = info.cross_verified; j["cross_verified"] = info.cross_verified;
j["keys"] = info.keys; j["keys"] = info.keys;
@ -2894,13 +2893,12 @@ to_json(json &j, const UserCache &info)
void void
from_json(const json &j, UserCache &info) from_json(const json &j, UserCache &info)
{ {
info.user_id = j.at("user_id");
info.is_user_verified = j.at("is_user_verified"); info.is_user_verified = j.at("is_user_verified");
info.cross_verified = j.at("cross_verified").get<std::vector<std::string>>(); info.cross_verified = j.at("cross_verified").get<std::vector<std::string>>();
info.keys = j.at("keys").get<mtx::responses::QueryKeys>(); info.keys = j.at("keys").get<mtx::responses::QueryKeys>();
} }
UserCache std::optional<UserCache>
Cache::getUserCache(const std::string &user_id) Cache::getUserCache(const std::string &user_id)
{ {
lmdb::val verifiedVal; lmdb::val verifiedVal;
@ -2909,14 +2907,15 @@ Cache::getUserCache(const std::string &user_id)
auto db = getUserCacheDb(txn); auto db = getUserCacheDb(txn);
auto res = lmdb::dbi_get(txn, db, lmdb::val(user_id), verifiedVal); auto res = lmdb::dbi_get(txn, db, lmdb::val(user_id), verifiedVal);
txn.commit();
UserCache verified_state; UserCache verified_state;
if (res) { if (res) {
verified_state = json::parse(std::string(verifiedVal.data(), verifiedVal.size())); verified_state = json::parse(std::string(verifiedVal.data(), verifiedVal.size()));
}
txn.commit();
return verified_state; return verified_state;
} else {
return {};
}
} }
//! be careful when using make sure is_user_verified is not changed //! be careful when using make sure is_user_verified is not changed
@ -2948,18 +2947,18 @@ Cache::deleteUserCache(const std::string &user_id)
void void
to_json(json &j, const DeviceVerifiedCache &info) to_json(json &j, const DeviceVerifiedCache &info)
{ {
j["user_id"] = info.user_id;
j["device_verified"] = info.device_verified; j["device_verified"] = info.device_verified;
j["device_blocked"] = info.device_blocked;
} }
void void
from_json(const json &j, DeviceVerifiedCache &info) from_json(const json &j, DeviceVerifiedCache &info)
{ {
info.user_id = j.at("user_id");
info.device_verified = j.at("device_verified").get<std::vector<std::string>>(); info.device_verified = j.at("device_verified").get<std::vector<std::string>>();
info.device_blocked = j.at("device_blocked").get<std::vector<std::string>>();
} }
DeviceVerifiedCache std::optional<DeviceVerifiedCache>
Cache::getVerifiedCache(const std::string &user_id) Cache::getVerifiedCache(const std::string &user_id)
{ {
lmdb::val verifiedVal; lmdb::val verifiedVal;
@ -2968,14 +2967,15 @@ Cache::getVerifiedCache(const std::string &user_id)
auto db = getDeviceVerifiedDb(txn); auto db = getDeviceVerifiedDb(txn);
auto res = lmdb::dbi_get(txn, db, lmdb::val(user_id), verifiedVal); auto res = lmdb::dbi_get(txn, db, lmdb::val(user_id), verifiedVal);
txn.commit();
DeviceVerifiedCache verified_state; DeviceVerifiedCache verified_state;
if (res) { if (res) {
verified_state = json::parse(std::string(verifiedVal.data(), verifiedVal.size())); verified_state = json::parse(std::string(verifiedVal.data(), verifiedVal.size()));
}
txn.commit();
return verified_state; return verified_state;
} else {
return {};
}
} }
int int
@ -3172,7 +3172,7 @@ statusMessage(const std::string &user_id)
{ {
return instance_->statusMessage(user_id); return instance_->statusMessage(user_id);
} }
UserCache std::optional<UserCache>
getUserCache(const std::string &user_id) getUserCache(const std::string &user_id)
{ {
return instance_->getUserCache(user_id); return instance_->getUserCache(user_id);
@ -3190,7 +3190,7 @@ deleteUserCache(const std::string &user_id)
return instance_->deleteUserCache(user_id); return instance_->deleteUserCache(user_id);
} }
DeviceVerifiedCache std::optional<DeviceVerifiedCache>
getVerifiedCache(const std::string &user_id) getVerifiedCache(const std::string &user_id)
{ {
return instance_->getVerifiedCache(user_id); return instance_->getVerifiedCache(user_id);

View file

@ -61,7 +61,7 @@ std::string
statusMessage(const std::string &user_id); statusMessage(const std::string &user_id);
//! user Cache //! user Cache
UserCache std::optional<UserCache>
getUserCache(const std::string &user_id); getUserCache(const std::string &user_id);
int int
@ -71,7 +71,7 @@ int
deleteUserCache(const std::string &user_id); deleteUserCache(const std::string &user_id);
//! verified Cache //! verified Cache
DeviceVerifiedCache std::optional<DeviceVerifiedCache>
getVerifiedCache(const std::string &user_id); getVerifiedCache(const std::string &user_id);
int int

View file

@ -68,8 +68,6 @@ struct OlmSessionStorage
struct UserCache struct UserCache
{ {
//! user_id of the user
std::string user_id;
//! this stores if the user is verified (with cross-signing) //! this stores if the user is verified (with cross-signing)
bool is_user_verified = false; bool is_user_verified = false;
//! list of verified device_ids with cross-signing //! list of verified device_ids with cross-signing
@ -85,10 +83,9 @@ from_json(const nlohmann::json &j, UserCache &info);
struct DeviceVerifiedCache struct DeviceVerifiedCache
{ {
//! user_id of the user
std::string user_id;
//! list of verified device_ids with device-verification //! list of verified device_ids with device-verification
std::vector<std::string> device_verified; std::vector<std::string> device_verified;
std::vector<std::string> device_blocked;
}; };
void void

View file

@ -55,12 +55,12 @@ public:
std::string statusMessage(const std::string &user_id); std::string statusMessage(const std::string &user_id);
// user cache stores user keys // user cache stores user keys
UserCache getUserCache(const std::string &user_id); std::optional<UserCache> getUserCache(const std::string &user_id);
int setUserCache(const std::string &user_id, const UserCache &body); int setUserCache(const std::string &user_id, const UserCache &body);
int deleteUserCache(const std::string &user_id); int deleteUserCache(const std::string &user_id);
// device verified cache // device verified cache
DeviceVerifiedCache getVerifiedCache(const std::string &user_id); std::optional<DeviceVerifiedCache> getVerifiedCache(const std::string &user_id);
int setVerifiedCache(const std::string &user_id, const DeviceVerifiedCache &body); int setVerifiedCache(const std::string &user_id, const DeviceVerifiedCache &body);
static void removeDisplayName(const QString &room_id, const QString &user_id); static void removeDisplayName(const QString &room_id, const QString &user_id);

View file

@ -7,11 +7,25 @@
#include <iostream> // only for debugging #include <iostream> // only for debugging
Q_DECLARE_METATYPE(UserProfile::Status)
UserProfile::UserProfile(QObject *parent) UserProfile::UserProfile(QObject *parent)
: QObject(parent) : QObject(parent)
{} {
qRegisterMetaType<UserProfile::Status>();
connect(
this, &UserProfile::updateDeviceList, this, [this]() { fetchDeviceList(this->userId); });
connect(
this,
&UserProfile::appendDeviceList,
this,
[this](QString device_id, QString device_name, UserProfile::Status verification_status) {
this->deviceList.push_back(
DeviceInfo{device_id, device_name, verification_status});
});
}
QVector<DeviceInfo> std::vector<DeviceInfo>
UserProfile::getDeviceList() UserProfile::getDeviceList()
{ {
return this->deviceList; return this->deviceList;
@ -37,7 +51,8 @@ UserProfile::setUserId(const QString &user_id)
void void
UserProfile::callback_fn(const mtx::responses::QueryKeys &res, UserProfile::callback_fn(const mtx::responses::QueryKeys &res,
mtx::http::RequestErr err, mtx::http::RequestErr err,
std::string user_id) std::string user_id,
std::optional<std::vector<std::string>> cross_verified)
{ {
if (err) { if (err) {
nhlog::net()->warn("failed to query device keys: {},{}", nhlog::net()->warn("failed to query device keys: {},{}",
@ -52,24 +67,40 @@ UserProfile::callback_fn(const mtx::responses::QueryKeys &res,
} }
auto devices = res.device_keys.at(user_id); auto devices = res.device_keys.at(user_id);
QVector<DeviceInfo> deviceInfo; std::vector<DeviceInfo> deviceInfo;
auto device_verified = cache::getVerifiedCache(user_id);
for (const auto &d : devices) { for (const auto &d : devices) {
auto device = d.second; auto device = d.second;
// TODO: Verify signatures and ignore those that don't pass. // TODO: Verify signatures and ignore those that don't pass.
DeviceInfo newdevice( UserProfile::Status verified = UserProfile::Status::UNVERIFIED;
QString::fromStdString(d.first), if (cross_verified.has_value()) {
QString::fromStdString(device.unsigned_info.device_display_name)); if (std::find(cross_verified->begin(), cross_verified->end(), d.first) !=
QString::fromStdString(device.unsigned_info.device_display_name); cross_verified->end())
verified = UserProfile::Status::VERIFIED;
deviceInfo.append(std::move(newdevice)); } else if (device_verified.has_value()) {
if (std::find(device_verified->device_verified.begin(),
device_verified->device_verified.end(),
d.first) != device_verified->device_verified.end())
verified = UserProfile::Status::VERIFIED;
} else if (device_verified.has_value()) {
if (std::find(device_verified->device_blocked.begin(),
device_verified->device_blocked.end(),
d.first) != device_verified->device_blocked.end())
verified = UserProfile::Status::BLOCKED;
} }
std::sort( emit UserProfile::appendDeviceList(
deviceInfo.begin(), deviceInfo.end(), [](const DeviceInfo &a, const DeviceInfo &b) { QString::fromStdString(d.first),
return a.device_id > b.device_id; QString::fromStdString(device.unsigned_info.device_display_name),
}); verified);
}
// std::sort(
// deviceInfo.begin(), deviceInfo.end(), [](const DeviceInfo &a, const DeviceInfo &b) {
// return a.device_id > b.device_id;
// });
this->deviceList = std::move(deviceInfo); this->deviceList = std::move(deviceInfo);
emit UserProfile::deviceListUpdated(); emit UserProfile::deviceListUpdated();
@ -81,9 +112,9 @@ UserProfile::fetchDeviceList(const QString &userID)
auto localUser = utils::localUser(); auto localUser = utils::localUser();
auto user_cache = cache::getUserCache(userID.toStdString()); auto user_cache = cache::getUserCache(userID.toStdString());
if (user_cache.user_id == userID.toStdString()) { if (user_cache.has_value()) {
mtx::http::ClientError error; this->callback_fn(
this->callback_fn(user_cache.keys, std::move(error), userID.toStdString()); user_cache->keys, {}, userID.toStdString(), user_cache->cross_verified);
} else { } else {
mtx::requests::QueryKeys req; mtx::requests::QueryKeys req;
req.device_keys[userID.toStdString()] = {}; req.device_keys[userID.toStdString()] = {};
@ -91,17 +122,11 @@ UserProfile::fetchDeviceList(const QString &userID)
req, req,
[user_id = userID.toStdString(), this](const mtx::responses::QueryKeys &res, [user_id = userID.toStdString(), this](const mtx::responses::QueryKeys &res,
mtx::http::RequestErr err) { mtx::http::RequestErr err) {
this->callback_fn(res, err, user_id); this->callback_fn(res, err, user_id, {});
}); });
} }
} }
void
UserProfile::updateDeviceList()
{
fetchDeviceList(this->userId);
}
void void
UserProfile::banUser() UserProfile::banUser()
{ {

View file

@ -5,36 +5,32 @@
#include <QVector> #include <QVector>
#include "MatrixClient.h" #include "MatrixClient.h"
class DeviceInfo
{
public:
DeviceInfo(const QString deviceID, const QString displayName)
: device_id(deviceID)
, display_name(displayName)
{}
DeviceInfo() {} class DeviceInfo;
QString device_id;
QString display_name;
};
class UserProfile : public QObject class UserProfile : public QObject
{ {
Q_OBJECT Q_OBJECT
Q_PROPERTY(QString userId READ getUserId WRITE setUserId NOTIFY userIdChanged) Q_PROPERTY(QString userId READ getUserId WRITE setUserId NOTIFY userIdChanged)
Q_PROPERTY(QVector<DeviceInfo> deviceList READ getDeviceList NOTIFY deviceListUpdated) Q_PROPERTY(std::vector<DeviceInfo> deviceList READ getDeviceList NOTIFY deviceListUpdated)
public: public:
// constructor // constructor
explicit UserProfile(QObject *parent = 0); explicit UserProfile(QObject *parent = 0);
// getters // getters
QVector<DeviceInfo> getDeviceList(); std::vector<DeviceInfo> getDeviceList();
QString getUserId(); QString getUserId();
// setters // setters
void setUserId(const QString &userId); void setUserId(const QString &userId);
Q_INVOKABLE void fetchDeviceList(const QString &userID); enum Status
Q_INVOKABLE void updateDeviceList(); {
VERIFIED,
UNVERIFIED,
BLOCKED
};
Q_ENUM(Status)
void fetchDeviceList(const QString &userID);
Q_INVOKABLE void banUser(); Q_INVOKABLE void banUser();
// Q_INVOKABLE void ignoreUser(); // Q_INVOKABLE void ignoreUser();
Q_INVOKABLE void kickUser(); Q_INVOKABLE void kickUser();
@ -43,12 +39,34 @@ public:
signals: signals:
void userIdChanged(); void userIdChanged();
void deviceListUpdated(); void deviceListUpdated();
void updateDeviceList();
void appendDeviceList(const QString device_id,
const QString device_naem,
const UserProfile::Status verification_status);
private: private:
QVector<DeviceInfo> deviceList; std::vector<DeviceInfo> deviceList;
QString userId; QString userId;
std::optional<std::string> cross_verified;
void callback_fn(const mtx::responses::QueryKeys &res, void callback_fn(const mtx::responses::QueryKeys &res,
mtx::http::RequestErr err, mtx::http::RequestErr err,
std::string user_id); std::string user_id,
std::optional<std::vector<std::string>> cross_verified);
};
class DeviceInfo
{
public:
DeviceInfo(const QString deviceID,
const QString displayName,
UserProfile::Status verification_status_)
: device_id(deviceID)
, display_name(displayName)
, verification_status(verification_status_)
{}
QString device_id;
QString display_name;
UserProfile::Status verification_status;
}; };

View file

@ -1,11 +1,23 @@
#include "UserProfileModel.h" #include "UserProfileModel.h"
#include "UserProfile.h"
#include <QModelIndex> #include <QModelIndex>
UserProfileModel::UserProfileModel(QObject *parent) UserProfileModel::UserProfileModel(QObject *parent)
: QAbstractListModel(parent) : QAbstractListModel(parent)
, deviceList(nullptr) , deviceList(nullptr)
{} {
this->deviceList = new UserProfile(this);
connect(this->deviceList, &UserProfile::userIdChanged, this, [this]() {
emit this->deviceList->updateDeviceList();
});
connect(this->deviceList, &UserProfile::deviceListUpdated, this, [this]() {
beginResetModel();
this->beginInsertRows(
QModelIndex(), 0, this->deviceList->getDeviceList().size() - 1);
this->endInsertRows();
endResetModel();
});
}
int int
UserProfileModel::rowCount(const QModelIndex &parent) const UserProfileModel::rowCount(const QModelIndex &parent) const
@ -18,7 +30,8 @@ UserProfileModel::rowCount(const QModelIndex &parent) const
QVariant QVariant
UserProfileModel::data(const QModelIndex &index, int role) const UserProfileModel::data(const QModelIndex &index, int role) const
{ {
if (!index.isValid() || !this->deviceList) if (!index.isValid() &&
static_cast<int>(this->deviceList->getDeviceList().size()) <= index.row())
return QVariant(); return QVariant();
const DeviceInfo device = this->deviceList->getDeviceList().at(index.row()); const DeviceInfo device = this->deviceList->getDeviceList().at(index.row());
@ -27,6 +40,8 @@ UserProfileModel::data(const QModelIndex &index, int role) const
return QVariant(device.device_id); return QVariant(device.device_id);
case DISPLAYNAME: case DISPLAYNAME:
return QVariant(device.display_name); return QVariant(device.display_name);
case VERIFIED_STATUS:
return device.verification_status;
} }
return QVariant(); return QVariant();
} }
@ -37,6 +52,7 @@ UserProfileModel::roleNames() const
QHash<int, QByteArray> names; QHash<int, QByteArray> names;
names[DEVICEID] = "deviceID"; names[DEVICEID] = "deviceID";
names[DISPLAYNAME] = "displayName"; names[DISPLAYNAME] = "displayName";
names[VERIFIED_STATUS] = "verified_status";
return names; return names;
} }
@ -45,22 +61,3 @@ UserProfileModel::getList() const
{ {
return (this->deviceList); return (this->deviceList);
} }
void
UserProfileModel::setList(UserProfile *devices)
{
beginResetModel();
if (devices)
devices->disconnect(this);
if (this->deviceList) {
const int index = this->deviceList->getDeviceList().size();
beginInsertRows(QModelIndex(), index, index);
endInsertRows();
}
this->deviceList = devices;
endResetModel();
}

View file

@ -1,5 +1,6 @@
#pragma once #pragma once
#include "UserProfile.h"
#include <QAbstractListModel> #include <QAbstractListModel>
class UserProfile; // forward declaration of the class UserProfile class UserProfile; // forward declaration of the class UserProfile
@ -7,7 +8,7 @@ class UserProfile; // forward declaration of the class UserProfile
class UserProfileModel : public QAbstractListModel class UserProfileModel : public QAbstractListModel
{ {
Q_OBJECT Q_OBJECT
Q_PROPERTY(UserProfile *deviceList READ getList WRITE setList) Q_PROPERTY(UserProfile *deviceList READ getList)
public: public:
explicit UserProfileModel(QObject *parent = nullptr); explicit UserProfileModel(QObject *parent = nullptr);
@ -15,10 +16,10 @@ public:
enum enum
{ {
DEVICEID, DEVICEID,
DISPLAYNAME DISPLAYNAME,
VERIFIED_STATUS
}; };
UserProfile *getList() const; UserProfile *getList() const;
void setList(UserProfile *devices);
int rowCount(const QModelIndex &parent = QModelIndex()) const override; int rowCount(const QModelIndex &parent = QModelIndex()) const override;
QVariant data(const QModelIndex &index, int role) const override; QVariant data(const QModelIndex &index, int role) const override;