1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
|
// Copyright 2005-2019 The Mumble Developers. All rights reserved.
// Use of this source code is governed by a BSD-style license
// that can be found in the LICENSE file at the root of the
// Mumble source tree or at <https://www.mumble.info/LICENSE>.
#include "mumble_pch.hpp"
#include "WebFetch.h"
#include "Global.h"
#include "NetworkConfig.h"
WebFetch::WebFetch(QString service, QUrl url, QObject *obj, const char *slot)
: QObject()
, qoObject(obj)
, cpSlot(slot)
, m_service(service) {
url.setScheme(QLatin1String("https"));
if (!g.s.qsServicePrefix.isEmpty()) {
url.setHost(prefixedServiceHost());
} else {
url.setHost(serviceHost());
}
qnr = Network::get(url);
connect(qnr, SIGNAL(finished()), this, SLOT(finished()));
connect(this, SIGNAL(fetched(QByteArray,QUrl,QMap<QString,QString>)), obj, slot);
}
QString WebFetch::prefixedServiceHost() const {
if (g.s.qsServicePrefix.isEmpty()) {
return serviceHost();
}
return QString::fromLatin1("%1-%2.mumble.info").arg(g.s.qsServicePrefix, m_service);
}
QString WebFetch::serviceHost() const {
return QString::fromLatin1("%1.mumble.info").arg(m_service);
}
static QString fromUtf8(const QByteArray &qba) {
if (qba.isEmpty())
return QString();
return QString::fromUtf8(qba.constData(), qba.length());
}
void WebFetch::finished() {
// Note that if this functions succeeds, it should deleteLater() itself, as this is a temporary object.
Q_ASSERT(qobject_cast<QNetworkReply *>(sender()) == qnr);
qnr->disconnect();
qnr->deleteLater();
QUrl url = qnr->request().url();
if (qnr->error() == QNetworkReply::NoError) {
QByteArray a = qnr->readAll();
// empty response is not an error
if (a.isNull())
a.append("");
QMap<QString, QString> headers;
foreach(const QByteArray &headerName, qnr->rawHeaderList()) {
QString name = fromUtf8(headerName);
QString value = fromUtf8(qnr->rawHeader(headerName));
if (! name.isEmpty() && ! value.isEmpty()) {
headers.insert(name, value);
if (name == QLatin1String("Use-Service-Prefix")) {
QRegExp servicePrefixRegExp(QLatin1String("^[a-zA-Z]+$"));
if (servicePrefixRegExp.exactMatch(value)) {
g.s.qsServicePrefix = value.toLower();
}
}
}
}
emit fetched(a, url, headers);
deleteLater();
} else if (url.host() == prefixedServiceHost()) {
url.setHost(serviceHost());
qnr = Network::get(url);
connect(qnr, SIGNAL(finished()), this, SLOT(finished()));
} else {
emit fetched(QByteArray(), url, QMap<QString,QString>());
deleteLater();
}
}
/**
* @brief Fetch URL from mumble servers.
*
* If fetching fails, the slot is invoked with a null QByteArray.
* @param url URL to fetch. Hostname and scheme must be blank.
* @param obj Object to invoke slot on.
* @param slot Slot to be triggered, invoked with the signature of \link fetched.
*/
void WebFetch::fetch(const QString &service, const QUrl &url, QObject *obj, const char *slot) {
Q_ASSERT(!service.isEmpty());
Q_ASSERT(url.scheme().isEmpty());
Q_ASSERT(url.host().isEmpty());
Q_ASSERT(obj);
Q_ASSERT(slot);
new WebFetch(service, url, obj, slot);
}
|