2017-04-06 02:06:42 +03:00
|
|
|
/*
|
|
|
|
* nheko Copyright (C) 2017 Konstantinos Sideris <siderisk@auth.gr>
|
|
|
|
*
|
|
|
|
* 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, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <QDebug>
|
2017-09-10 12:58:00 +03:00
|
|
|
#include <QFile>
|
|
|
|
#include <QImageReader>
|
2017-04-27 00:32:33 +03:00
|
|
|
#include <QJsonArray>
|
2017-04-06 02:06:42 +03:00
|
|
|
#include <QJsonDocument>
|
|
|
|
#include <QJsonObject>
|
2017-11-30 00:39:35 +03:00
|
|
|
#include <QMimeDatabase>
|
2017-04-06 02:06:42 +03:00
|
|
|
#include <QNetworkReply>
|
|
|
|
#include <QNetworkRequest>
|
2017-04-11 17:45:47 +03:00
|
|
|
#include <QPixmap>
|
2017-04-06 02:06:42 +03:00
|
|
|
#include <QSettings>
|
|
|
|
#include <QUrlQuery>
|
|
|
|
|
|
|
|
#include "Login.h"
|
|
|
|
#include "MatrixClient.h"
|
2017-04-08 02:53:23 +03:00
|
|
|
#include "Register.h"
|
2017-04-06 02:06:42 +03:00
|
|
|
|
|
|
|
MatrixClient::MatrixClient(QString server, QObject *parent)
|
2017-08-20 13:47:22 +03:00
|
|
|
: QNetworkAccessManager(parent)
|
2017-11-06 00:04:55 +03:00
|
|
|
, clientApiUrl_{"/_matrix/client/r0"}
|
|
|
|
, mediaApiUrl_{"/_matrix/media/r0"}
|
|
|
|
, server_{"https://" + server}
|
2017-04-06 02:06:42 +03:00
|
|
|
{
|
2017-09-03 11:43:45 +03:00
|
|
|
QSettings settings;
|
|
|
|
txn_id_ = settings.value("client/transaction_id", 1).toInt();
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2018-01-11 17:33:50 +03:00
|
|
|
QJsonObject default_filter{
|
2018-01-11 17:34:43 +03:00
|
|
|
{
|
|
|
|
"room",
|
|
|
|
QJsonObject{
|
|
|
|
{"include_leave", true},
|
|
|
|
{
|
|
|
|
"account_data",
|
|
|
|
QJsonObject{
|
|
|
|
{"not_types", QJsonArray{"*"}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"account_data",
|
2018-01-11 17:33:50 +03:00
|
|
|
QJsonObject{
|
|
|
|
{"not_types", QJsonArray{"*"}},
|
|
|
|
},
|
2018-01-11 17:34:43 +03:00
|
|
|
},
|
|
|
|
{
|
|
|
|
"presence",
|
2018-01-11 17:33:50 +03:00
|
|
|
QJsonObject{
|
|
|
|
{"not_types", QJsonArray{"*"}},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2018-01-11 17:34:43 +03:00
|
|
|
filter_ = settings
|
|
|
|
.value("client/sync_filter",
|
|
|
|
QJsonDocument(default_filter).toJson(QJsonDocument::Compact))
|
|
|
|
.toString();
|
2018-01-11 17:33:50 +03:00
|
|
|
|
2017-10-05 18:13:11 +03:00
|
|
|
connect(this,
|
|
|
|
&QNetworkAccessManager::networkAccessibleChanged,
|
|
|
|
this,
|
|
|
|
[=](NetworkAccessibility status) {
|
|
|
|
if (status != NetworkAccessibility::Accessible)
|
|
|
|
setNetworkAccessible(NetworkAccessibility::Accessible);
|
|
|
|
});
|
2017-04-06 02:06:42 +03:00
|
|
|
}
|
|
|
|
|
2017-08-20 13:47:22 +03:00
|
|
|
void
|
|
|
|
MatrixClient::reset() noexcept
|
2017-04-09 02:17:04 +03:00
|
|
|
{
|
2017-10-21 21:17:01 +03:00
|
|
|
next_batch_.clear();
|
|
|
|
server_.clear();
|
|
|
|
token_.clear();
|
2017-04-09 02:17:04 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
txn_id_ = 0;
|
2017-04-09 02:17:04 +03:00
|
|
|
}
|
|
|
|
|
2017-08-20 13:47:22 +03:00
|
|
|
void
|
2017-10-22 22:51:50 +03:00
|
|
|
MatrixClient::login(const QString &username, const QString &password) noexcept
|
2017-10-01 19:49:36 +03:00
|
|
|
{
|
2017-10-22 22:51:50 +03:00
|
|
|
QUrl endpoint(server_);
|
|
|
|
endpoint.setPath(clientApiUrl_ + "/login");
|
2017-10-08 22:38:38 +03:00
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
QNetworkRequest request(endpoint);
|
|
|
|
request.setHeader(QNetworkRequest::ContentTypeHeader, "application/json");
|
2017-10-08 22:38:38 +03:00
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
LoginRequest body(username, password);
|
2017-10-01 19:49:36 +03:00
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
auto reply = post(request, body.serialize());
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply]() {
|
|
|
|
reply->deleteLater();
|
2017-10-08 22:38:38 +03:00
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
int status_code =
|
|
|
|
reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
2017-10-01 19:49:36 +03:00
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
if (status_code == 403) {
|
|
|
|
emit loginError(tr("Wrong username or password"));
|
|
|
|
return;
|
|
|
|
}
|
2017-10-01 19:49:36 +03:00
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
if (status_code == 404) {
|
|
|
|
emit loginError(tr("Login endpoint was not found on the server"));
|
|
|
|
return;
|
|
|
|
}
|
2017-10-01 19:49:36 +03:00
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
if (status_code >= 400) {
|
|
|
|
qWarning() << "Login error: " << reply->errorString();
|
|
|
|
emit loginError(tr("An unknown error occured. Please try again."));
|
|
|
|
return;
|
|
|
|
}
|
2017-10-01 19:49:36 +03:00
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
try {
|
2017-12-03 03:47:37 +03:00
|
|
|
mtx::responses::Login login =
|
|
|
|
nlohmann::json::parse(reply->readAll().data());
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
auto hostname = server_.host();
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
if (server_.port() > 0)
|
|
|
|
hostname = QString("%1:%2").arg(server_.host()).arg(server_.port());
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-12-03 03:47:37 +03:00
|
|
|
emit loginSuccess(QString::fromStdString(login.user_id.toString()),
|
|
|
|
hostname,
|
|
|
|
QString::fromStdString(login.access_token));
|
2017-12-03 03:50:46 +03:00
|
|
|
} catch (std::exception &e) {
|
2017-10-22 22:51:50 +03:00
|
|
|
qWarning() << "Malformed JSON response" << e.what();
|
|
|
|
emit loginError(tr("Malformed response. Possibly not a Matrix server"));
|
|
|
|
}
|
|
|
|
});
|
2017-04-06 02:06:42 +03:00
|
|
|
}
|
2017-08-20 13:47:22 +03:00
|
|
|
void
|
|
|
|
MatrixClient::logout() noexcept
|
2017-04-09 02:17:04 +03:00
|
|
|
{
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("access_token", token_);
|
2017-04-09 02:17:04 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrl endpoint(server_);
|
2017-09-10 12:58:00 +03:00
|
|
|
endpoint.setPath(clientApiUrl_ + "/logout");
|
2017-09-03 11:43:45 +03:00
|
|
|
endpoint.setQuery(query);
|
2017-04-09 02:17:04 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QNetworkRequest request(endpoint);
|
|
|
|
request.setHeader(QNetworkRequest::ContentTypeHeader, "application/json");
|
2017-04-09 02:17:04 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QJsonObject body{};
|
2017-10-22 22:51:50 +03:00
|
|
|
auto reply = post(request, QJsonDocument(body).toJson(QJsonDocument::Compact));
|
|
|
|
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status != 200) {
|
|
|
|
qWarning() << "Logout error: " << reply->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
emit loggedOut();
|
|
|
|
});
|
2017-04-09 02:17:04 +03:00
|
|
|
}
|
|
|
|
|
2017-08-20 13:47:22 +03:00
|
|
|
void
|
|
|
|
MatrixClient::registerUser(const QString &user, const QString &pass, const QString &server) noexcept
|
2017-04-08 02:53:23 +03:00
|
|
|
{
|
2017-09-03 11:43:45 +03:00
|
|
|
setServer(server);
|
2017-04-08 02:53:23 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("kind", "user");
|
2017-04-08 02:53:23 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrl endpoint(server_);
|
2017-09-10 12:58:00 +03:00
|
|
|
endpoint.setPath(clientApiUrl_ + "/register");
|
2017-09-03 11:43:45 +03:00
|
|
|
endpoint.setQuery(query);
|
2017-04-08 02:53:23 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QNetworkRequest request(QString(endpoint.toEncoded()));
|
|
|
|
request.setHeader(QNetworkRequest::ContentTypeHeader, "application/json");
|
2017-04-08 02:53:23 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
RegisterRequest body(user, pass);
|
2017-10-22 22:51:50 +03:00
|
|
|
auto reply = post(request, body.serialize());
|
|
|
|
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
auto data = reply->readAll();
|
|
|
|
auto json = QJsonDocument::fromJson(data);
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
if (json.isObject() && json.object().contains("error"))
|
|
|
|
emit registerError(json.object().value("error").toString());
|
|
|
|
else
|
|
|
|
emit registerError(reply->errorString());
|
2017-04-08 02:53:23 +03:00
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
RegisterResponse response;
|
|
|
|
|
|
|
|
try {
|
|
|
|
response.deserialize(json);
|
|
|
|
emit registerSuccess(response.getUserId(),
|
|
|
|
response.getHomeServer(),
|
|
|
|
response.getAccessToken());
|
|
|
|
} catch (DeserializationException &e) {
|
|
|
|
qWarning() << "Register" << e.what();
|
|
|
|
emit registerError("Received malformed response.");
|
|
|
|
}
|
|
|
|
});
|
2017-04-08 02:53:23 +03:00
|
|
|
}
|
|
|
|
|
2017-08-20 13:47:22 +03:00
|
|
|
void
|
|
|
|
MatrixClient::sync() noexcept
|
2017-04-06 02:06:42 +03:00
|
|
|
{
|
2018-01-13 15:49:51 +03:00
|
|
|
// the filter is not uploaded yet (so it is a json with { at the beginning)
|
|
|
|
// ignore for now that the filter might be uploaded multiple times as we expect
|
|
|
|
// servers to do deduplication
|
|
|
|
if (filter_.startsWith("{")) {
|
|
|
|
uploadFilter(filter_);
|
|
|
|
}
|
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("set_presence", "online");
|
2018-01-11 17:33:50 +03:00
|
|
|
query.addQueryItem("filter", filter_);
|
|
|
|
query.addQueryItem("timeout", "30000");
|
2017-09-03 11:43:45 +03:00
|
|
|
query.addQueryItem("access_token", token_);
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
if (next_batch_.isEmpty()) {
|
2017-10-01 12:11:33 +03:00
|
|
|
qDebug() << "Sync requires a valid next_batch token. Initial sync should "
|
|
|
|
"be performed.";
|
2017-09-03 11:43:45 +03:00
|
|
|
return;
|
|
|
|
}
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
query.addQueryItem("since", next_batch_);
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrl endpoint(server_);
|
2017-09-10 12:58:00 +03:00
|
|
|
endpoint.setPath(clientApiUrl_ + "/sync");
|
2017-09-03 11:43:45 +03:00
|
|
|
endpoint.setQuery(query);
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QNetworkRequest request(QString(endpoint.toEncoded()));
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
auto reply = get(request);
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
emit syncFailed(reply->errorString());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
2017-12-04 19:41:19 +03:00
|
|
|
mtx::responses::Sync response = nlohmann::json::parse(reply->readAll());
|
2017-10-22 22:51:50 +03:00
|
|
|
emit syncCompleted(response);
|
2017-12-04 19:41:19 +03:00
|
|
|
} catch (std::exception &e) {
|
2017-12-16 20:31:07 +03:00
|
|
|
qWarning() << "Sync malformed response: " << e.what();
|
2017-10-22 22:51:50 +03:00
|
|
|
}
|
|
|
|
});
|
2017-04-06 02:06:42 +03:00
|
|
|
}
|
|
|
|
|
2017-08-20 13:47:22 +03:00
|
|
|
void
|
2017-12-04 19:41:19 +03:00
|
|
|
MatrixClient::sendRoomMessage(mtx::events::MessageType ty,
|
2017-11-15 19:38:50 +03:00
|
|
|
int txnId,
|
2017-09-03 11:43:45 +03:00
|
|
|
const QString &roomid,
|
2017-09-10 12:58:00 +03:00
|
|
|
const QString &msg,
|
2017-12-01 18:33:49 +03:00
|
|
|
const QFileInfo &fileinfo,
|
2017-09-10 12:58:00 +03:00
|
|
|
const QString &url) noexcept
|
2017-04-06 02:06:42 +03:00
|
|
|
{
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("access_token", token_);
|
|
|
|
|
|
|
|
QUrl endpoint(server_);
|
2017-09-10 12:58:00 +03:00
|
|
|
endpoint.setPath(clientApiUrl_ +
|
2017-11-15 19:38:50 +03:00
|
|
|
QString("/rooms/%1/send/m.room.message/%2").arg(roomid).arg(txnId));
|
2017-09-03 11:43:45 +03:00
|
|
|
endpoint.setQuery(query);
|
|
|
|
|
|
|
|
QString msgType("");
|
2017-12-01 18:33:49 +03:00
|
|
|
|
|
|
|
QMimeDatabase db;
|
|
|
|
QMimeType mime =
|
|
|
|
db.mimeTypeForFile(fileinfo.absoluteFilePath(), QMimeDatabase::MatchContent);
|
|
|
|
|
2017-09-10 12:58:00 +03:00
|
|
|
QJsonObject body;
|
2017-12-01 18:33:49 +03:00
|
|
|
QJsonObject info = {{"size", fileinfo.size()}, {"mimetype", mime.name()}};
|
2017-09-03 11:43:45 +03:00
|
|
|
|
|
|
|
switch (ty) {
|
2017-12-04 19:41:19 +03:00
|
|
|
case mtx::events::MessageType::Text:
|
2017-11-06 00:04:55 +03:00
|
|
|
body = {{"msgtype", "m.text"}, {"body", msg}};
|
2017-09-03 11:43:45 +03:00
|
|
|
break;
|
2017-12-04 19:41:19 +03:00
|
|
|
case mtx::events::MessageType::Emote:
|
2017-11-06 00:04:55 +03:00
|
|
|
body = {{"msgtype", "m.emote"}, {"body", msg}};
|
2017-09-03 11:43:45 +03:00
|
|
|
break;
|
2017-12-04 19:41:19 +03:00
|
|
|
case mtx::events::MessageType::Image:
|
2017-12-01 18:33:49 +03:00
|
|
|
body = {{"msgtype", "m.image"}, {"body", msg}, {"url", url}, {"info", info}};
|
2017-09-03 11:43:45 +03:00
|
|
|
break;
|
2017-12-04 19:41:19 +03:00
|
|
|
case mtx::events::MessageType::File:
|
2017-12-01 18:33:49 +03:00
|
|
|
body = {{"msgtype", "m.file"}, {"body", msg}, {"url", url}, {"info", info}};
|
|
|
|
break;
|
2017-12-04 19:41:19 +03:00
|
|
|
case mtx::events::MessageType::Audio:
|
2017-12-01 18:33:49 +03:00
|
|
|
body = {{"msgtype", "m.audio"}, {"body", msg}, {"url", url}, {"info", info}};
|
2017-11-30 00:39:35 +03:00
|
|
|
break;
|
2017-09-10 12:58:00 +03:00
|
|
|
default:
|
|
|
|
qDebug() << "SendRoomMessage: Unknown message type for" << msg;
|
|
|
|
return;
|
2017-09-03 11:43:45 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
QNetworkRequest request(QString(endpoint.toEncoded()));
|
|
|
|
request.setHeader(QNetworkRequest::ContentTypeHeader, "application/json");
|
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
auto reply = put(request, QJsonDocument(body).toJson(QJsonDocument::Compact));
|
|
|
|
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply, roomid, txnId]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
2017-11-15 19:38:50 +03:00
|
|
|
emit messageSendFailed(roomid, txnId);
|
2017-10-22 22:51:50 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto data = reply->readAll();
|
|
|
|
|
2017-11-15 19:38:50 +03:00
|
|
|
if (data.isEmpty()) {
|
|
|
|
emit messageSendFailed(roomid, txnId);
|
2017-10-22 22:51:50 +03:00
|
|
|
return;
|
2017-11-15 19:38:50 +03:00
|
|
|
}
|
2017-10-22 22:51:50 +03:00
|
|
|
|
|
|
|
auto json = QJsonDocument::fromJson(data);
|
|
|
|
|
|
|
|
if (!json.isObject()) {
|
|
|
|
qDebug() << "Send message response is not a JSON object";
|
2017-11-15 19:38:50 +03:00
|
|
|
emit messageSendFailed(roomid, txnId);
|
2017-10-22 22:51:50 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto object = json.object();
|
|
|
|
|
|
|
|
if (!object.contains("event_id")) {
|
|
|
|
qDebug() << "SendTextMessage: missing event_id from response";
|
2017-11-15 19:38:50 +03:00
|
|
|
emit messageSendFailed(roomid, txnId);
|
2017-10-22 22:51:50 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
emit messageSent(object.value("event_id").toString(), roomid, txnId);
|
|
|
|
});
|
2017-04-06 02:06:42 +03:00
|
|
|
}
|
|
|
|
|
2017-08-20 13:47:22 +03:00
|
|
|
void
|
|
|
|
MatrixClient::initialSync() noexcept
|
2017-04-06 02:06:42 +03:00
|
|
|
{
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrlQuery query;
|
2017-10-05 08:47:29 +03:00
|
|
|
query.addQueryItem("timeout", "0");
|
2018-01-11 17:33:50 +03:00
|
|
|
query.addQueryItem("filter", filter_);
|
2017-09-03 11:43:45 +03:00
|
|
|
query.addQueryItem("access_token", token_);
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrl endpoint(server_);
|
2017-09-10 12:58:00 +03:00
|
|
|
endpoint.setPath(clientApiUrl_ + "/sync");
|
2017-09-03 11:43:45 +03:00
|
|
|
endpoint.setQuery(query);
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QNetworkRequest request(QString(endpoint.toEncoded()));
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
auto reply = get(request);
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
emit initialSyncFailed(reply->errorString());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
2017-12-04 19:41:19 +03:00
|
|
|
mtx::responses::Sync response = nlohmann::json::parse(reply->readAll());
|
|
|
|
emit initialSyncCompleted(response);
|
2017-12-06 22:25:56 +03:00
|
|
|
} catch (std::exception &e) {
|
2017-10-22 22:51:50 +03:00
|
|
|
qWarning() << "Sync malformed response" << e.what();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
2017-04-06 02:06:42 +03:00
|
|
|
}
|
|
|
|
|
2017-08-20 13:47:22 +03:00
|
|
|
void
|
|
|
|
MatrixClient::versions() noexcept
|
2017-04-06 02:06:42 +03:00
|
|
|
{
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrl endpoint(server_);
|
|
|
|
endpoint.setPath("/_matrix/client/versions");
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QNetworkRequest request(endpoint);
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
auto reply = get(request);
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status_code =
|
|
|
|
reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status_code == 404) {
|
|
|
|
emit versionError("Versions endpoint was not found on the server. Possibly "
|
|
|
|
"not a Matrix server");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (status_code >= 400) {
|
|
|
|
emit versionError("An unknown error occured. Please try again.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
2017-12-03 03:47:37 +03:00
|
|
|
mtx::responses::Versions versions =
|
|
|
|
nlohmann::json::parse(reply->readAll().data());
|
|
|
|
|
|
|
|
emit versionSuccess();
|
|
|
|
} catch (std::exception &e) {
|
2017-10-22 22:51:50 +03:00
|
|
|
emit versionError("Malformed response. Possibly not a Matrix server");
|
|
|
|
}
|
|
|
|
});
|
2017-04-06 02:06:42 +03:00
|
|
|
}
|
|
|
|
|
2017-08-20 13:47:22 +03:00
|
|
|
void
|
|
|
|
MatrixClient::getOwnProfile() noexcept
|
2017-04-06 02:06:42 +03:00
|
|
|
{
|
2017-10-01 12:11:33 +03:00
|
|
|
// FIXME: Remove settings from the matrix client. The class should store the
|
|
|
|
// user's matrix ID.
|
2017-09-03 11:43:45 +03:00
|
|
|
QSettings settings;
|
|
|
|
auto userid = settings.value("auth/user_id", "").toString();
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("access_token", token_);
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrl endpoint(server_);
|
2017-09-10 12:58:00 +03:00
|
|
|
endpoint.setPath(clientApiUrl_ + "/profile/" + userid);
|
2017-09-03 11:43:45 +03:00
|
|
|
endpoint.setQuery(query);
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QNetworkRequest request(QString(endpoint.toEncoded()));
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QNetworkReply *reply = get(request);
|
2017-10-22 22:51:50 +03:00
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status >= 400) {
|
|
|
|
qWarning() << reply->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
2017-12-03 03:47:37 +03:00
|
|
|
mtx::responses::Profile profile =
|
|
|
|
nlohmann::json::parse(reply->readAll().data());
|
|
|
|
|
|
|
|
emit getOwnProfileResponse(QUrl(QString::fromStdString(profile.avatar_url)),
|
|
|
|
QString::fromStdString(profile.display_name));
|
|
|
|
} catch (std::exception &e) {
|
2017-10-22 22:51:50 +03:00
|
|
|
qWarning() << "Profile:" << e.what();
|
|
|
|
}
|
|
|
|
});
|
2017-04-06 02:06:42 +03:00
|
|
|
}
|
2017-04-11 17:45:47 +03:00
|
|
|
|
2018-01-09 16:07:32 +03:00
|
|
|
void
|
|
|
|
MatrixClient::getOwnCommunities() noexcept
|
|
|
|
{
|
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("access_token", token_);
|
|
|
|
|
|
|
|
QUrl endpoint(server_);
|
|
|
|
endpoint.setPath(clientApiUrl_ + "/joined_groups");
|
|
|
|
endpoint.setQuery(query);
|
|
|
|
|
|
|
|
QNetworkRequest request(QString(endpoint.toEncoded()));
|
|
|
|
|
|
|
|
QNetworkReply *reply = get(request);
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status >= 400) {
|
|
|
|
qWarning() << reply->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto data = reply->readAll();
|
|
|
|
auto json = QJsonDocument::fromJson(data).object();
|
|
|
|
|
|
|
|
try {
|
|
|
|
QList<QString> response;
|
|
|
|
for (auto it = json["groups"].toArray().constBegin();
|
|
|
|
it != json["groups"].toArray().constEnd();
|
|
|
|
it++) {
|
|
|
|
response.append(it->toString());
|
|
|
|
}
|
|
|
|
emit getOwnCommunitiesResponse(response);
|
|
|
|
} catch (DeserializationException &e) {
|
|
|
|
qWarning() << "Own communities:" << e.what();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-08-20 13:47:22 +03:00
|
|
|
void
|
|
|
|
MatrixClient::fetchRoomAvatar(const QString &roomid, const QUrl &avatar_url)
|
2017-04-11 17:45:47 +03:00
|
|
|
{
|
2017-09-03 11:43:45 +03:00
|
|
|
QList<QString> url_parts = avatar_url.toString().split("mxc://");
|
2017-04-11 17:45:47 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
if (url_parts.size() != 2) {
|
|
|
|
qDebug() << "Invalid format for room avatar " << avatar_url.toString();
|
|
|
|
return;
|
|
|
|
}
|
2017-04-11 17:45:47 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("width", "512");
|
|
|
|
query.addQueryItem("height", "512");
|
|
|
|
query.addQueryItem("method", "crop");
|
2017-05-27 01:29:45 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QString media_url =
|
|
|
|
QString("%1/_matrix/media/r0/thumbnail/%2").arg(getHomeServer().toString(), url_parts[1]);
|
2017-05-27 01:29:45 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrl endpoint(media_url);
|
|
|
|
endpoint.setQuery(query);
|
2017-04-11 17:45:47 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QNetworkRequest avatar_request(endpoint);
|
2017-04-11 17:45:47 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QNetworkReply *reply = get(avatar_request);
|
2017-12-22 01:00:48 +03:00
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply, roomid, avatar_url]() {
|
2017-10-22 22:51:50 +03:00
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
qWarning() << reply->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto img = reply->readAll();
|
|
|
|
|
|
|
|
if (img.size() == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
QPixmap pixmap;
|
|
|
|
pixmap.loadFromData(img);
|
|
|
|
|
2017-12-22 01:00:48 +03:00
|
|
|
emit roomAvatarRetrieved(roomid, pixmap, avatar_url.toString(), img);
|
2017-10-22 22:51:50 +03:00
|
|
|
});
|
2017-04-11 17:45:47 +03:00
|
|
|
}
|
|
|
|
|
2018-01-09 16:07:32 +03:00
|
|
|
void
|
|
|
|
MatrixClient::fetchCommunityAvatar(const QString &communityId, const QUrl &avatar_url)
|
|
|
|
{
|
|
|
|
QList<QString> url_parts = avatar_url.toString().split("mxc://");
|
|
|
|
|
|
|
|
if (url_parts.size() != 2) {
|
|
|
|
qDebug() << "Invalid format for community avatar " << avatar_url.toString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("width", "512");
|
|
|
|
query.addQueryItem("height", "512");
|
|
|
|
query.addQueryItem("method", "crop");
|
|
|
|
|
|
|
|
QString media_url =
|
|
|
|
QString("%1/_matrix/media/r0/thumbnail/%2").arg(getHomeServer().toString(), url_parts[1]);
|
|
|
|
|
|
|
|
QUrl endpoint(media_url);
|
|
|
|
endpoint.setQuery(query);
|
|
|
|
|
|
|
|
QNetworkRequest avatar_request(endpoint);
|
|
|
|
|
|
|
|
QNetworkReply *reply = get(avatar_request);
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply, communityId]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
qWarning() << reply->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto img = reply->readAll();
|
|
|
|
|
|
|
|
if (img.size() == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
QPixmap pixmap;
|
|
|
|
pixmap.loadFromData(img);
|
|
|
|
|
|
|
|
emit communityAvatarRetrieved(communityId, pixmap);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
MatrixClient::fetchCommunityProfile(const QString &communityId)
|
|
|
|
{
|
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("access_token", token_);
|
|
|
|
|
|
|
|
QUrl endpoint(server_);
|
|
|
|
endpoint.setPath(clientApiUrl_ + "/groups/" + communityId + "/profile");
|
|
|
|
endpoint.setQuery(query);
|
|
|
|
|
|
|
|
QNetworkRequest request(QString(endpoint.toEncoded()));
|
|
|
|
|
|
|
|
QNetworkReply *reply = get(request);
|
|
|
|
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply, communityId]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status >= 400) {
|
|
|
|
qWarning() << reply->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto data = reply->readAll();
|
|
|
|
const auto json = QJsonDocument::fromJson(data).object();
|
|
|
|
|
|
|
|
emit communityProfileRetrieved(communityId, json);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
MatrixClient::fetchCommunityRooms(const QString &communityId)
|
|
|
|
{
|
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("access_token", token_);
|
|
|
|
|
|
|
|
QUrl endpoint(server_);
|
|
|
|
endpoint.setPath(clientApiUrl_ + "/groups/" + communityId + "/rooms");
|
|
|
|
endpoint.setQuery(query);
|
|
|
|
|
|
|
|
QNetworkRequest request(QString(endpoint.toEncoded()));
|
|
|
|
|
|
|
|
QNetworkReply *reply = get(request);
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply, communityId]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status >= 400) {
|
|
|
|
qWarning() << reply->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto data = reply->readAll();
|
|
|
|
const auto json = QJsonDocument::fromJson(data).object();
|
|
|
|
|
|
|
|
emit communityRoomsRetrieved(communityId, json);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-08-20 13:47:22 +03:00
|
|
|
void
|
|
|
|
MatrixClient::fetchUserAvatar(const QString &userId, const QUrl &avatarUrl)
|
2017-06-05 02:14:05 +03:00
|
|
|
{
|
2017-09-03 11:43:45 +03:00
|
|
|
QList<QString> url_parts = avatarUrl.toString().split("mxc://");
|
2017-06-05 02:14:05 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
if (url_parts.size() != 2) {
|
|
|
|
qDebug() << "Invalid format for user avatar " << avatarUrl.toString();
|
|
|
|
return;
|
|
|
|
}
|
2017-06-05 02:14:05 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("width", "128");
|
|
|
|
query.addQueryItem("height", "128");
|
|
|
|
query.addQueryItem("method", "crop");
|
2017-06-05 02:14:05 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QString media_url =
|
|
|
|
QString("%1/_matrix/media/r0/thumbnail/%2").arg(getHomeServer().toString(), url_parts[1]);
|
2017-06-05 02:14:05 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrl endpoint(media_url);
|
|
|
|
endpoint.setQuery(query);
|
2017-06-05 02:14:05 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QNetworkRequest avatar_request(endpoint);
|
2017-06-05 02:14:05 +03:00
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
auto reply = get(avatar_request);
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply, userId]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
qWarning() << reply->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto data = reply->readAll();
|
|
|
|
|
|
|
|
if (data.size() == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
QImage img;
|
|
|
|
img.loadFromData(data);
|
|
|
|
|
|
|
|
emit userAvatarRetrieved(userId, img);
|
|
|
|
});
|
2017-06-05 02:14:05 +03:00
|
|
|
}
|
|
|
|
|
2017-08-20 13:47:22 +03:00
|
|
|
void
|
|
|
|
MatrixClient::downloadImage(const QString &event_id, const QUrl &url)
|
2017-04-28 14:56:45 +03:00
|
|
|
{
|
2017-09-03 11:43:45 +03:00
|
|
|
QNetworkRequest image_request(url);
|
2017-04-28 14:56:45 +03:00
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
auto reply = get(image_request);
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply, event_id]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
qWarning() << reply->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto img = reply->readAll();
|
|
|
|
|
|
|
|
if (img.size() == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
QPixmap pixmap;
|
|
|
|
pixmap.loadFromData(img);
|
|
|
|
|
|
|
|
emit imageDownloaded(event_id, pixmap);
|
|
|
|
});
|
2017-04-28 14:56:45 +03:00
|
|
|
}
|
|
|
|
|
2017-11-28 03:01:37 +03:00
|
|
|
void
|
|
|
|
MatrixClient::downloadFile(const QString &event_id, const QUrl &url)
|
|
|
|
{
|
|
|
|
QNetworkRequest fileRequest(url);
|
|
|
|
|
|
|
|
auto reply = get(fileRequest);
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply, event_id]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
// TODO: Handle error
|
|
|
|
qWarning() << reply->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto data = reply->readAll();
|
|
|
|
|
|
|
|
if (data.size() == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
emit fileDownloaded(event_id, data);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-08-20 13:47:22 +03:00
|
|
|
void
|
|
|
|
MatrixClient::fetchOwnAvatar(const QUrl &avatar_url)
|
2017-04-11 17:45:47 +03:00
|
|
|
{
|
2017-09-03 11:43:45 +03:00
|
|
|
QList<QString> url_parts = avatar_url.toString().split("mxc://");
|
2017-04-11 17:45:47 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
if (url_parts.size() != 2) {
|
|
|
|
qDebug() << "Invalid format for media " << avatar_url.toString();
|
|
|
|
return;
|
|
|
|
}
|
2017-04-11 17:45:47 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("width", "512");
|
|
|
|
query.addQueryItem("height", "512");
|
|
|
|
query.addQueryItem("method", "crop");
|
2017-05-27 01:29:45 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QString media_url =
|
|
|
|
QString("%1/_matrix/media/r0/thumbnail/%2").arg(getHomeServer().toString(), url_parts[1]);
|
2017-05-27 01:29:45 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrl endpoint(media_url);
|
|
|
|
endpoint.setQuery(query);
|
2017-04-11 17:45:47 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QNetworkRequest avatar_request(endpoint);
|
2017-04-11 17:45:47 +03:00
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
auto reply = get(avatar_request);
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
qWarning() << reply->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto img = reply->readAll();
|
|
|
|
|
|
|
|
if (img.size() == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
QPixmap pixmap;
|
|
|
|
pixmap.loadFromData(img);
|
|
|
|
|
|
|
|
emit ownAvatarRetrieved(pixmap);
|
|
|
|
});
|
2017-04-11 17:45:47 +03:00
|
|
|
}
|
2017-05-12 15:43:35 +03:00
|
|
|
|
2017-08-20 13:47:22 +03:00
|
|
|
void
|
2017-10-22 22:51:50 +03:00
|
|
|
MatrixClient::messages(const QString &roomid, const QString &from_token, int limit) noexcept
|
2017-05-12 15:43:35 +03:00
|
|
|
{
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("access_token", token_);
|
|
|
|
query.addQueryItem("from", from_token);
|
|
|
|
query.addQueryItem("dir", "b");
|
|
|
|
query.addQueryItem("limit", QString::number(limit));
|
2017-05-12 15:43:35 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QUrl endpoint(server_);
|
2017-10-22 22:51:50 +03:00
|
|
|
endpoint.setPath(clientApiUrl_ + QString("/rooms/%1/messages").arg(roomid));
|
2017-09-03 11:43:45 +03:00
|
|
|
endpoint.setQuery(query);
|
2017-05-12 15:43:35 +03:00
|
|
|
|
2017-09-03 11:43:45 +03:00
|
|
|
QNetworkRequest request(QString(endpoint.toEncoded()));
|
2017-05-12 15:43:35 +03:00
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
auto reply = get(request);
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply, roomid]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
qWarning() << reply->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
2017-12-04 19:41:19 +03:00
|
|
|
mtx::responses::Messages messages =
|
|
|
|
nlohmann::json::parse(reply->readAll().data());
|
|
|
|
|
|
|
|
emit messagesRetrieved(roomid, messages);
|
|
|
|
} catch (std::exception &e) {
|
2017-10-22 22:51:50 +03:00
|
|
|
qWarning() << "Room messages from" << roomid << e.what();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
});
|
2017-05-12 15:43:35 +03:00
|
|
|
}
|
2017-09-10 12:58:00 +03:00
|
|
|
|
|
|
|
void
|
2018-01-10 10:52:59 +03:00
|
|
|
MatrixClient::uploadImage(const QString &roomid,
|
|
|
|
const QSharedPointer<QIODevice> data,
|
|
|
|
const QString &filename)
|
2017-09-10 12:58:00 +03:00
|
|
|
{
|
2018-01-10 10:52:59 +03:00
|
|
|
auto reply = makeUploadRequest(data);
|
2017-09-10 12:58:00 +03:00
|
|
|
|
2017-12-01 18:33:49 +03:00
|
|
|
if (reply == nullptr)
|
2017-09-10 12:58:00 +03:00
|
|
|
return;
|
|
|
|
|
2018-01-10 10:52:59 +03:00
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply, roomid, data, filename]() {
|
2017-10-22 22:51:50 +03:00
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
emit syncFailed(reply->errorString());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-01-10 10:52:59 +03:00
|
|
|
auto res_data = reply->readAll();
|
2017-10-22 22:51:50 +03:00
|
|
|
|
2018-01-10 10:52:59 +03:00
|
|
|
if (res_data.isEmpty())
|
2017-10-22 22:51:50 +03:00
|
|
|
return;
|
|
|
|
|
2018-01-10 10:52:59 +03:00
|
|
|
auto json = QJsonDocument::fromJson(res_data);
|
2017-10-22 22:51:50 +03:00
|
|
|
|
|
|
|
if (!json.isObject()) {
|
|
|
|
qDebug() << "Media upload: Response is not a json object.";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QJsonObject object = json.object();
|
|
|
|
if (!object.contains("content_uri")) {
|
|
|
|
qDebug() << "Media upload: Missing content_uri key";
|
|
|
|
qDebug() << object;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-01-10 10:52:59 +03:00
|
|
|
emit imageUploaded(roomid, data, filename, object.value("content_uri").toString());
|
2017-10-22 22:51:50 +03:00
|
|
|
});
|
2017-09-10 12:58:00 +03:00
|
|
|
}
|
2017-10-01 19:49:36 +03:00
|
|
|
|
2017-11-30 00:39:35 +03:00
|
|
|
void
|
2018-01-10 10:52:59 +03:00
|
|
|
MatrixClient::uploadFile(const QString &roomid,
|
|
|
|
const QSharedPointer<QIODevice> data,
|
|
|
|
const QString &filename)
|
2017-11-30 00:39:35 +03:00
|
|
|
{
|
2018-01-10 10:52:59 +03:00
|
|
|
auto reply = makeUploadRequest(data);
|
2017-11-30 00:39:35 +03:00
|
|
|
|
2018-01-10 10:52:59 +03:00
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply, roomid, data, filename]() {
|
2017-12-01 18:33:49 +03:00
|
|
|
reply->deleteLater();
|
2017-11-30 00:39:35 +03:00
|
|
|
|
2017-12-01 18:33:49 +03:00
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
2017-11-30 00:39:35 +03:00
|
|
|
|
2017-12-01 18:33:49 +03:00
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
emit syncFailed(reply->errorString());
|
|
|
|
return;
|
|
|
|
}
|
2017-11-30 00:39:35 +03:00
|
|
|
|
2017-12-01 18:33:49 +03:00
|
|
|
auto data = reply->readAll();
|
|
|
|
|
|
|
|
if (data.isEmpty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
auto json = QJsonDocument::fromJson(data);
|
|
|
|
|
|
|
|
if (!json.isObject()) {
|
|
|
|
qDebug() << "Media upload: Response is not a json object.";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QJsonObject object = json.object();
|
|
|
|
if (!object.contains("content_uri")) {
|
|
|
|
qDebug() << "Media upload: Missing content_uri key";
|
|
|
|
qDebug() << object;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
emit fileUploaded(roomid, filename, object.value("content_uri").toString());
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2018-01-10 10:52:59 +03:00
|
|
|
MatrixClient::uploadAudio(const QString &roomid,
|
|
|
|
const QSharedPointer<QIODevice> data,
|
|
|
|
const QString &filename)
|
2017-12-01 18:33:49 +03:00
|
|
|
{
|
2018-01-10 10:52:59 +03:00
|
|
|
auto reply = makeUploadRequest(data);
|
2017-11-30 00:39:35 +03:00
|
|
|
|
2018-01-10 10:52:59 +03:00
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply, roomid, data, filename]() {
|
2017-11-30 00:39:35 +03:00
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
emit syncFailed(reply->errorString());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto data = reply->readAll();
|
|
|
|
|
|
|
|
if (data.isEmpty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
auto json = QJsonDocument::fromJson(data);
|
|
|
|
|
|
|
|
if (!json.isObject()) {
|
|
|
|
qDebug() << "Media upload: Response is not a json object.";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QJsonObject object = json.object();
|
|
|
|
if (!object.contains("content_uri")) {
|
|
|
|
qDebug() << "Media upload: Missing content_uri key";
|
|
|
|
qDebug() << object;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-12-01 18:33:49 +03:00
|
|
|
emit audioUploaded(roomid, filename, object.value("content_uri").toString());
|
2017-11-30 00:39:35 +03:00
|
|
|
});
|
2018-01-13 15:49:51 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
MatrixClient::uploadFilter(const QString &filter) noexcept
|
|
|
|
{
|
|
|
|
// validate that filter is a Json-String
|
|
|
|
QJsonDocument doc = QJsonDocument::fromJson(filter.toUtf8());
|
|
|
|
if (doc.isNull() || !doc.isObject()) {
|
|
|
|
qWarning() << "Input which should be uploaded as filter is no JsonObject";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QSettings settings;
|
|
|
|
auto userid = settings.value("auth/user_id", "").toString();
|
|
|
|
|
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("access_token", token_);
|
|
|
|
|
|
|
|
QUrl endpoint(server_);
|
|
|
|
endpoint.setPath(clientApiUrl_ + QString("/user/%1/filter").arg(userid));
|
|
|
|
endpoint.setQuery(query);
|
|
|
|
|
|
|
|
QNetworkRequest request(endpoint);
|
|
|
|
request.setHeader(QNetworkRequest::KnownHeaders::ContentTypeHeader, "application/json");
|
|
|
|
|
|
|
|
auto reply = post(request, doc.toJson(QJsonDocument::Compact));
|
|
|
|
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
qWarning() << reply->errorString() << "42";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto data = reply->readAll();
|
|
|
|
auto response = QJsonDocument::fromJson(data);
|
|
|
|
auto filter_id = response.object()["filter_id"].toString();
|
|
|
|
|
|
|
|
qDebug() << "Filter with ID" << filter_id << "created.";
|
|
|
|
QSettings settings;
|
|
|
|
settings.setValue("client/sync_filter", filter_id);
|
|
|
|
settings.sync();
|
|
|
|
|
|
|
|
// set the filter_ var so following syncs will use it
|
|
|
|
filter_ = filter_id;
|
|
|
|
});
|
2017-11-30 00:39:35 +03:00
|
|
|
}
|
2017-12-01 18:33:49 +03:00
|
|
|
|
2017-10-01 19:49:36 +03:00
|
|
|
void
|
|
|
|
MatrixClient::joinRoom(const QString &roomIdOrAlias)
|
|
|
|
{
|
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("access_token", token_);
|
|
|
|
|
|
|
|
QUrl endpoint(server_);
|
|
|
|
endpoint.setPath(clientApiUrl_ + QString("/join/%1").arg(roomIdOrAlias));
|
|
|
|
endpoint.setQuery(query);
|
|
|
|
|
|
|
|
QNetworkRequest request(endpoint);
|
|
|
|
request.setHeader(QNetworkRequest::KnownHeaders::ContentTypeHeader, "application/json");
|
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
auto reply = post(request, "{}");
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
auto data = reply->readAll();
|
|
|
|
auto response = QJsonDocument::fromJson(data);
|
|
|
|
auto json = response.object();
|
|
|
|
|
|
|
|
if (json.contains("error"))
|
|
|
|
emit joinFailed(json["error"].toString());
|
|
|
|
else
|
|
|
|
qDebug() << reply->errorString();
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto data = reply->readAll();
|
|
|
|
auto response = QJsonDocument::fromJson(data);
|
|
|
|
auto room_id = response.object()["room_id"].toString();
|
|
|
|
|
|
|
|
emit joinedRoom(room_id);
|
|
|
|
});
|
2017-10-01 19:49:36 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
MatrixClient::leaveRoom(const QString &roomId)
|
|
|
|
{
|
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("access_token", token_);
|
|
|
|
|
|
|
|
QUrl endpoint(server_);
|
|
|
|
endpoint.setPath(clientApiUrl_ + QString("/rooms/%1/leave").arg(roomId));
|
|
|
|
endpoint.setQuery(query);
|
|
|
|
|
|
|
|
QNetworkRequest request(endpoint);
|
|
|
|
request.setHeader(QNetworkRequest::KnownHeaders::ContentTypeHeader, "application/json");
|
|
|
|
|
2017-10-22 22:51:50 +03:00
|
|
|
auto reply = post(request, "{}");
|
|
|
|
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply, roomId]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
qWarning() << reply->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
emit leftRoom(roomId);
|
|
|
|
});
|
2017-10-01 19:49:36 +03:00
|
|
|
}
|
2017-10-31 21:11:49 +03:00
|
|
|
|
2017-12-11 00:59:50 +03:00
|
|
|
void
|
|
|
|
MatrixClient::inviteUser(const QString &roomId, const QString &user)
|
|
|
|
{
|
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("access_token", token_);
|
|
|
|
|
|
|
|
QUrl endpoint(server_);
|
|
|
|
endpoint.setPath(clientApiUrl_ + QString("/rooms/%1/invite").arg(roomId));
|
|
|
|
endpoint.setQuery(query);
|
|
|
|
|
|
|
|
QNetworkRequest request(endpoint);
|
|
|
|
request.setHeader(QNetworkRequest::KnownHeaders::ContentTypeHeader, "application/json");
|
|
|
|
|
|
|
|
QJsonObject body{{"user_id", user}};
|
|
|
|
auto reply = post(request, QJsonDocument(body).toJson(QJsonDocument::Compact));
|
|
|
|
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply, roomId, user]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
// TODO: Handle failure.
|
|
|
|
qWarning() << reply->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
emit invitedUser(roomId, user);
|
|
|
|
});
|
|
|
|
}
|
2017-12-12 00:00:37 +03:00
|
|
|
|
|
|
|
void
|
|
|
|
MatrixClient::createRoom(const mtx::requests::CreateRoom &create_room_request)
|
|
|
|
{
|
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("access_token", token_);
|
|
|
|
|
|
|
|
QUrl endpoint(server_);
|
|
|
|
endpoint.setPath(clientApiUrl_ + QString("/createRoom"));
|
|
|
|
endpoint.setQuery(query);
|
|
|
|
|
|
|
|
QNetworkRequest request(endpoint);
|
|
|
|
request.setHeader(QNetworkRequest::KnownHeaders::ContentTypeHeader, "application/json");
|
|
|
|
|
|
|
|
nlohmann::json body = create_room_request;
|
|
|
|
auto reply = post(request, QString::fromStdString(body.dump()).toUtf8());
|
|
|
|
|
|
|
|
connect(reply, &QNetworkReply::finished, this, [this, reply]() {
|
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
auto data = reply->readAll();
|
|
|
|
auto response = QJsonDocument::fromJson(data);
|
|
|
|
auto json = response.object();
|
|
|
|
|
|
|
|
if (json.contains("error"))
|
|
|
|
emit roomCreationFailed(json["error"].toString());
|
|
|
|
else
|
|
|
|
qDebug() << reply->errorString();
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto data = reply->readAll();
|
|
|
|
auto response = QJsonDocument::fromJson(data);
|
|
|
|
auto room_id = response.object()["room_id"].toString();
|
|
|
|
|
|
|
|
emit roomCreated(room_id);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-10-31 21:11:49 +03:00
|
|
|
void
|
|
|
|
MatrixClient::sendTypingNotification(const QString &roomid, int timeoutInMillis)
|
|
|
|
{
|
|
|
|
QSettings settings;
|
|
|
|
QString user_id = settings.value("auth/user_id").toString();
|
|
|
|
|
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("access_token", token_);
|
|
|
|
|
|
|
|
QUrl endpoint(server_);
|
|
|
|
endpoint.setPath(clientApiUrl_ + QString("/rooms/%1/typing/%2").arg(roomid).arg(user_id));
|
|
|
|
|
|
|
|
endpoint.setQuery(query);
|
|
|
|
|
|
|
|
QString msgType("");
|
|
|
|
QJsonObject body;
|
|
|
|
|
2017-11-06 00:04:55 +03:00
|
|
|
body = {{"typing", true}, {"timeout", timeoutInMillis}};
|
2017-10-31 21:11:49 +03:00
|
|
|
|
|
|
|
QNetworkRequest request(QString(endpoint.toEncoded()));
|
|
|
|
request.setHeader(QNetworkRequest::ContentTypeHeader, "application/json");
|
|
|
|
|
|
|
|
put(request, QJsonDocument(body).toJson(QJsonDocument::Compact));
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
MatrixClient::removeTypingNotification(const QString &roomid)
|
|
|
|
{
|
|
|
|
QSettings settings;
|
|
|
|
QString user_id = settings.value("auth/user_id").toString();
|
|
|
|
|
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("access_token", token_);
|
|
|
|
|
|
|
|
QUrl endpoint(server_);
|
|
|
|
endpoint.setPath(clientApiUrl_ + QString("/rooms/%1/typing/%2").arg(roomid).arg(user_id));
|
|
|
|
|
|
|
|
endpoint.setQuery(query);
|
|
|
|
|
|
|
|
QString msgType("");
|
|
|
|
QJsonObject body;
|
|
|
|
|
2017-11-06 00:04:55 +03:00
|
|
|
body = {{"typing", false}};
|
2017-10-31 21:11:49 +03:00
|
|
|
|
|
|
|
QNetworkRequest request(QString(endpoint.toEncoded()));
|
|
|
|
request.setHeader(QNetworkRequest::ContentTypeHeader, "application/json");
|
|
|
|
|
|
|
|
put(request, QJsonDocument(body).toJson(QJsonDocument::Compact));
|
|
|
|
}
|
2017-11-24 01:10:58 +03:00
|
|
|
|
|
|
|
void
|
|
|
|
MatrixClient::readEvent(const QString &room_id, const QString &event_id)
|
|
|
|
{
|
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("access_token", token_);
|
|
|
|
|
|
|
|
QUrl endpoint(server_);
|
|
|
|
endpoint.setPath(clientApiUrl_ +
|
|
|
|
QString("/rooms/%1/receipt/m.read/%2").arg(room_id).arg(event_id));
|
|
|
|
endpoint.setQuery(query);
|
|
|
|
|
|
|
|
QNetworkRequest request(QString(endpoint.toEncoded()));
|
|
|
|
request.setHeader(QNetworkRequest::KnownHeaders::ContentTypeHeader, "application/json");
|
|
|
|
|
|
|
|
auto reply = post(request, "{}");
|
|
|
|
|
2017-12-04 19:49:25 +03:00
|
|
|
connect(reply, &QNetworkReply::finished, this, [reply]() {
|
2017-11-24 01:10:58 +03:00
|
|
|
reply->deleteLater();
|
|
|
|
|
|
|
|
int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
|
|
|
if (status == 0 || status >= 400) {
|
|
|
|
qWarning() << reply->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2017-12-01 18:33:49 +03:00
|
|
|
|
|
|
|
QNetworkReply *
|
2018-01-10 10:52:59 +03:00
|
|
|
MatrixClient::makeUploadRequest(QSharedPointer<QIODevice> iodev)
|
2017-12-01 18:33:49 +03:00
|
|
|
{
|
|
|
|
QUrlQuery query;
|
|
|
|
query.addQueryItem("access_token", token_);
|
|
|
|
|
|
|
|
QUrl endpoint(server_);
|
|
|
|
endpoint.setPath(mediaApiUrl_ + "/upload");
|
|
|
|
endpoint.setQuery(query);
|
|
|
|
|
2018-01-10 10:52:59 +03:00
|
|
|
if (!iodev->open(QIODevice::ReadOnly)) {
|
|
|
|
qWarning() << "Error while reading device:" << iodev->errorString();
|
2017-12-01 18:33:49 +03:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
QMimeDatabase db;
|
2018-01-10 10:52:59 +03:00
|
|
|
QMimeType mime = db.mimeTypeForData(iodev.data());
|
2017-12-01 18:33:49 +03:00
|
|
|
|
|
|
|
QNetworkRequest request(QString(endpoint.toEncoded()));
|
|
|
|
request.setHeader(QNetworkRequest::ContentTypeHeader, mime.name());
|
|
|
|
|
2018-01-10 10:52:59 +03:00
|
|
|
auto reply = post(request, iodev.data());
|
2017-12-01 18:33:49 +03:00
|
|
|
|
|
|
|
return reply;
|
|
|
|
}
|