source: ntrip/trunk/BNC/src/upload/bncuploadcaster.cpp@ 9734

Last change on this file since 9734 was 9734, checked in by stuerze, 23 months ago

minor changes

File size: 11.5 KB
Line 
1/* -------------------------------------------------------------------------
2 * BKG NTRIP Server
3 * -------------------------------------------------------------------------
4 *
5 * Class: bncUploadCaster
6 *
7 * Purpose: Connection to NTRIP Caster
8 *
9 * Author: L. Mervart
10 *
11 * Created: 29-Mar-2011
12 *
13 * Changes:
14 *
15 * -----------------------------------------------------------------------*/
16
17#include <math.h>
18#include "bncuploadcaster.h"
19#include "bncversion.h"
20#include "bnccore.h"
21#include "bnctableitem.h"
22#include "bncsettings.h"
23#include "bncsslconfig.h"
24
25using namespace std;
26
27// Constructor
28////////////////////////////////////////////////////////////////////////////
29bncUploadCaster::bncUploadCaster(const QString &mountpoint,
30 const QString &outHost, int outPort, const QString &ntripVersion,
31 const QString &userName, const QString &password, int iRow, int rate) {
32 bncSettings settings;
33
34 _mountpoint = mountpoint;
35 _casterOutHost = outHost;
36 _casterOutPort = outPort;
37 _ntripVersion = ntripVersion;
38 _userName = userName;
39 _password = password;
40 _outSocket = 0;
41 _sOpenTrial = 0;
42 _iRow = iRow;
43 _rate = rate;
44
45 if (_rate < 0) {
46 _rate = 0;
47 } else if (_rate > 60) {
48 _rate = 60;
49 }
50 _isToBeDeleted = false;
51
52 connect(this, SIGNAL(newMessage(QByteArray,bool)), BNC_CORE, SLOT(slotMessage(const QByteArray,bool)));
53
54 if (BNC_CORE->_uploadTableItems.find(_iRow) != BNC_CORE->_uploadTableItems.end()) {
55 connect(this, SIGNAL(newBytes(QByteArray,double)),
56 BNC_CORE->_uploadTableItems.value(iRow),
57 SLOT(slotNewBytes(const QByteArray,double)));
58 }
59 if (BNC_CORE->_uploadEphTableItems.find(_iRow) != BNC_CORE->_uploadEphTableItems.end()) {
60 connect(this, SIGNAL(newBytes(QByteArray,double)),
61 BNC_CORE->_uploadEphTableItems.value(iRow),
62 SLOT(slotNewBytes(const QByteArray,double)));
63 }
64
65 _sslIgnoreErrors = (Qt::CheckState(settings.value("sslIgnoreErrors").toInt()) == Qt::Checked);
66
67 _proxyOutHost = settings.value("proxyHost").toString();
68 _proxyOutPort = settings.value("proxyPort").toInt();
69 (_proxyOutHost.isEmpty()) ? _proxy = false : _proxy = true;
70
71 _secure = false;
72 if (_ntripVersion == "2s") {
73 if (!QSslSocket::supportsSsl()) {
74 emit(newMessage(
75 "For SSL support please install OpenSSL run-time libraries: Ntrip Version 2 is tried",
76 true));
77 _ntripVersion == "2";
78 } else {
79 _secure = true;
80 _casterOutPort = 443;
81 }
82 }
83
84 if (!_secure && _proxy) {
85 _postExtension = QString("http://%1:%2").arg(_casterOutHost).arg(_casterOutPort);
86 } else {
87 _postExtension = "";
88 }
89}
90
91// Safe Desctructor
92////////////////////////////////////////////////////////////////////////////
93void bncUploadCaster::deleteSafely() {
94 _isToBeDeleted = true;
95 if (!isRunning()) {
96 delete this;
97 }
98}
99
100// Destructor
101////////////////////////////////////////////////////////////////////////////
102bncUploadCaster::~bncUploadCaster() {
103 if (isRunning()) {
104 wait();
105 }
106 if (_outSocket) {
107 disconnect(_outSocket, SIGNAL(sslErrors(QList<QSslError>)), this, SLOT(slotSslErrors(QList<QSslError>)));
108 disconnect(_outSocket, SIGNAL(disconnected()), this, SLOT(slotSslSocketConnectionClosed()));
109 delete _outSocket;
110 }
111}
112
113//
114////////////////////////////////////////////////////////////////////////////
115void bncUploadCaster::slotProxyAuthenticationRequired(const QNetworkProxy&,
116 QAuthenticator*) {
117 emit newMessage("slotProxyAuthenticationRequired", true);
118}
119
120// TSL/SSL
121 ////////////////////////////////////////////////////////////////////////////
122void bncUploadCaster::slotSslErrors(QList<QSslError> errors) {
123 emit(newMessage("slotSslErrors", true));
124 QString msg = "SSL Error: ";
125 if (_outSocket) {
126 QSslCertificate cert = _outSocket->sslConfiguration().peerCertificate();
127 if (!cert.isNull() &&
128 cert.issuerInfo(QSslCertificate::OrganizationalUnitName).count() &&
129 cert.issuerInfo(QSslCertificate::Organization).count()) {
130 msg += QString("Server Certificate Issued by:\n"
131 "%1\n%2\nCannot be verified\n")
132#if QT_VERSION >= 0x050000
133 .arg(cert.issuerInfo(QSslCertificate::OrganizationalUnitName).at(0))
134 .arg(cert.issuerInfo(QSslCertificate::Organization).at(0));
135#else
136 .arg(cert.issuerInfo(QSslCertificate::OrganizationalUnitName))
137 .arg(cert.issuerInfo(QSslCertificate::Organization));
138#endif
139 }
140
141 QListIterator<QSslError> it(errors);
142 while (it.hasNext()) {
143 const QSslError& err = it.next();
144 msg += err.errorString();
145 }
146 emit(newMessage(msg.toLatin1(), true));
147
148 if (_sslIgnoreErrors) {
149 _outSocket->ignoreSslErrors();
150 emit(newMessage("BNC ignores SSL errors as configured", true));
151 }
152 }
153}
154
155
156// Endless Loop
157////////////////////////////////////////////////////////////////////////////
158void bncUploadCaster::run() {
159 while (true) {
160 if (_isToBeDeleted) {
161 QThread::quit();
162 deleteLater();
163 return;
164 }
165 open();
166 if (_outSocket && _outSocket->state() == QAbstractSocket::ConnectedState) {
167 QMutexLocker locker(&_mutex);
168 if (_outBuffer.size() > 0) {
169 if (_ntripVersion == "1") {
170 _outSocket->write(_outBuffer);
171 _outSocket->flush();
172 } else {
173 QString chunkSize = QString("%1").arg(_outBuffer.size(), 0, 16, QLatin1Char('0'));
174 QByteArray chunkedData = chunkSize.toLatin1() + "\r\n" + _outBuffer + "\r\n";
175 _outSocket->write(chunkedData);
176 _outSocket->flush();
177 }
178 emit newBytes(_mountpoint.toLatin1(), _outBuffer.size());
179 }
180 }
181 if (_rate == 0) {
182 {
183 QMutexLocker locker(&_mutex);
184 _outBuffer.clear();
185 }
186 msleep(100); //sleep 0.1 sec
187 } else {
188 sleep(_rate);
189 }
190 }
191}
192
193// Start the Communication with NTRIP Caster
194////////////////////////////////////////////////////////////////////////////
195void bncUploadCaster::open() {
196 const int timeOut = 5000; // 5 seconds
197 bncSslConfig sslConfig;
198 QByteArray msg;
199
200 if (_mountpoint.isEmpty()) {
201 return;
202 }
203
204 if (_outSocket != 0) {
205 if (_outSocket->state() == QAbstractSocket::ConnectedState) {
206 return;
207 }
208 _outSocket->close();
209 }
210
211 double minDt = pow(2.0, _sOpenTrial);
212 if (++_sOpenTrial > 4) {
213 _sOpenTrial = 4;
214 }
215 if (_outSocketOpenTime.isValid()
216 && _outSocketOpenTime.secsTo(QDateTime::currentDateTime()) < minDt) {
217 return;
218 } else {
219 _outSocketOpenTime = QDateTime::currentDateTime();
220 }
221
222 _outSocket = new QSslSocket();
223 _outSocket->setProxy(QNetworkProxy::NoProxy);
224 _outSocket->setSslConfiguration(sslConfig);
225 connect(_outSocket, SIGNAL(disconnected()), this, SLOT(slotSslSocketConnectionClosed()));
226 connect(_outSocket, SIGNAL(sslErrors(QList<QSslError>)), this, SLOT(slotSslErrors(QList<QSslError>)));
227
228 if (!_proxy) {
229 if (!connectToHost(_casterOutHost, _casterOutPort, _secure)) {
230 return;
231 }
232 } else {
233 if (_ntripVersion == "1") {
234 emit(newMessage("No proxy support in Ntrip Version 1 upload!", true));
235 _outSocket->close();
236 return;
237 }
238 connect(_outSocket, SIGNAL(proxyAuthenticationRequired(const QNetworkProxy&, QAuthenticator*)),
239 this,SLOT(slotProxyAuthenticationRequired(const QNetworkProxy&, QAuthenticator*)));
240
241 if (!connectToHost(_proxyOutHost, _proxyOutPort, false)) {
242 return;
243 }
244
245 if (_secure) {
246 msg = "CONNECT " + _casterOutHost.toLatin1() + ":"
247 + QString("%1").arg(_casterOutPort).toLatin1() + " HTTP/1.1\r\n"
248 + "Proxy-Connection: Keep-Alive\r\n"
249 + "Host: " + _casterOutHost.toLatin1() + "\r\n"
250 + "User-Agent: NTRIP BNC/" BNCVERSION " (" + BNC_OS + ")\r\n"
251 + "\r\n"; //cout << msg.toStdString().c_str();
252 _outSocket->write(msg);
253 _outSocket->waitForBytesWritten();
254 _outSocket->waitForReadyRead();
255 QByteArray ans = _outSocket->readAll(); //cout << ans.toStdString().c_str() << endl;
256 if (ans.indexOf("200") == -1) {
257 emit(newMessage("Proxy: Connection broken for " + _mountpoint.toLatin1()
258 + ": " + ans.left(ans.length() - 2), true));
259 _outSocket->close();
260 return;
261 } else {
262 emit(newMessage("Proxy: Connection established for " + _mountpoint.toLatin1(), true));
263 _sOpenTrial = 0;
264 _outSocket->setPeerVerifyName(_casterOutHost);
265 _outSocket->startClientEncryption();
266 if (!_outSocket->waitForEncrypted(timeOut)) {
267 emit(newMessage("Proxy/Caster: Encrypt timeout for " + _mountpoint.toLatin1() + " ("
268 + _casterOutHost.toLatin1() + ":"
269 + QString("%1) ").arg(_casterOutPort).toLatin1()
270 + _outSocket->errorString().toLatin1(), true));
271 return;
272 } else {
273 emit(newMessage("Proxy: SSL handshake completed for " + _mountpoint.toLatin1(), true));
274 }
275 }
276 }
277 }
278
279 if (_ntripVersion == "1") {
280 msg = "SOURCE " + _password.toLatin1() + " /" + _mountpoint.toLatin1()
281 + "\r\n" + "Source-Agent: NTRIP BNC/" BNCVERSION "\r\n\r\n";
282 } else {
283 msg = "POST " + _postExtension.toLatin1() + "/" + _mountpoint.toLatin1()
284 + " HTTP/1.1\r\n" + "Host: " + _casterOutHost.toLatin1() + "\r\n"
285 + "Ntrip-Version: Ntrip/2.0\r\n" + "Authorization: Basic "
286 + (_userName + ":" + _password).toLatin1().toBase64() + "\r\n"
287 + "User-Agent: NTRIP BNC/" BNCVERSION " (" + BNC_OS + ")\r\n"
288 + "Connection: close\r\n" + "Transfer-Encoding: chunked\r\n\r\n";
289 }
290 //cout << msg.toStdString().c_str();
291 _outSocket->write(msg);
292 _outSocket->waitForBytesWritten();
293 _outSocket->waitForReadyRead();
294
295 QByteArray ans = "";
296 if (_outSocket->canReadLine()) {
297 ans = _outSocket->readLine(); //cout << "ans: " << ans.toStdString().c_str() << endl;
298 }
299
300 if (ans.indexOf("200") == -1) {
301 _outSocket->close();
302 emit(newMessage("Broadcaster: Connection broken for " + _mountpoint.toLatin1() + ": "
303 + ans.left(ans.length() - 2), true));
304 } else {
305 emit(newMessage("Broadcaster: Connection opened for " + _mountpoint.toLatin1(), true));
306 _sOpenTrial = 0;
307 }
308}
309
310// Try connection to NTRIP Caster or Proxy
311////////////////////////////////////////////////////////////////////////////
312bool bncUploadCaster::connectToHost(QString outHost, int outPort, bool encrypted) {
313 const int timeOut = 5000; // 5 seconds
314 if (encrypted) {
315 _outSocket->connectToHostEncrypted(outHost, outPort);
316 if (!_outSocket->waitForEncrypted(timeOut)) {
317 emit(newMessage(
318 "Broadcaster: Connect timeout for " + _mountpoint.toLatin1() + " ("
319 + outHost.toLatin1() + ":"
320 + QString("%1) ").arg(outPort).toLatin1()
321 + _outSocket->errorString().toLatin1(), true));
322 return false;
323 } else {
324 emit(newMessage("Broadcaster: SSL handshake completed for " + _mountpoint.toLatin1(), true));
325 }
326 } else {
327 _outSocket->connectToHost(outHost, outPort);
328 if (!_outSocket->waitForConnected(timeOut)) {
329 emit(newMessage(
330 "Broadcaster: Connect timeout for " + _mountpoint.toLatin1() + " ("
331 + outHost.toLatin1() + ":"
332 + QString("%1) ").arg(outPort).toLatin1()
333 + _outSocket->errorString().toLatin1(), true));
334 return false;
335 }
336 }
337 return true;
338}
339
340
341void bncUploadCaster::slotSslSocketConnectionClosed() {
342 disconnect(_outSocket, SIGNAL(sslErrors(QList<QSslError>)), this, SLOT(slotSslErrors(QList<QSslError>)));
343 emit(newMessage("slotSslSocketConnectionClosed", true));
344 if (_outSocket) {
345 delete _outSocket;
346 _outSocket = 0;
347 }
348}
349
350
351
352
Note: See TracBrowser for help on using the repository browser.