/**************************************************************************** ** ** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of the test suite of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 2.1 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3.0 as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU General Public License version 3.0 requirements will be ** met: http://www.gnu.org/copyleft/gpl.html. ** ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include #include class tst_QNetworkProxyFactory : public QObject { Q_OBJECT public: tst_QNetworkProxyFactory(); class QDebugProxyFactory : public QNetworkProxyFactory { public: virtual QList queryProxy(const QNetworkProxyQuery &query = QNetworkProxyQuery()) { returnedList = QNetworkProxyFactory::systemProxyForQuery(query); requestCounter++; return returnedList; } QList returnedList; static int requestCounter; }; private slots: void systemProxyForQueryCalledFromThread(); void systemProxyForQuery_data(); void systemProxyForQuery() const; void systemProxyForQuery_local(); #ifndef QT_NO_BEARERMANAGEMENT void fromConfigurations(); void inNetworkAccessManager_data(); void inNetworkAccessManager(); #endif private: QString formatProxyName(const QNetworkProxy & proxy) const; QDebugProxyFactory *factory; }; int tst_QNetworkProxyFactory::QDebugProxyFactory::requestCounter = 0; tst_QNetworkProxyFactory::tst_QNetworkProxyFactory() { factory = new QDebugProxyFactory; QNetworkProxyFactory::setApplicationProxyFactory(factory); } QString tst_QNetworkProxyFactory::formatProxyName(const QNetworkProxy & proxy) const { QString proxyName; if (!proxy.user().isNull()) proxyName.append("%1:%2@").arg(proxy.user(), proxy.password()); proxyName.append(QString("%1:%2").arg(proxy.hostName()).arg(proxy.port())); proxyName.append(QString(" (type=%1, capabilities=%2)").arg(proxy.type()).arg(proxy.capabilities())); return proxyName; } void tst_QNetworkProxyFactory::systemProxyForQuery_data() { QTest::addColumn("type"); QTest::addColumn("url"); QTest::addColumn("tag"); QTest::addColumn("hostName"); QTest::addColumn("port"); QTest::addColumn("requiredCapabilities"); //URLs QTest::newRow("http") << (int)QNetworkProxyQuery::UrlRequest << QUrl("http://qt-project.org") << QString() << QString() << 0 << 0; //windows: "intranet" should be bypassed if "bypass proxy server for local addresses" is ticked QTest::newRow("intranet") << (int)QNetworkProxyQuery::UrlRequest << QUrl("http://qt-test-server") << QString() << QString() << 0 << 0; //windows: "intranet2" should be bypassed if "*.local" is in the exceptions list (advanced settings) QTest::newRow("intranet2") << (int)QNetworkProxyQuery::UrlRequest << QUrl("http://qt-test-server.local") << QString() << QString() << 0 << 0; QTest::newRow("https") << (int)QNetworkProxyQuery::UrlRequest << QUrl("https://qt-project.org") << QString() << QString() << 0 << (int)QNetworkProxy::TunnelingCapability; QTest::newRow("ftp") << (int)QNetworkProxyQuery::UrlRequest << QUrl("ftp://qt-project.org") << QString() << QString() << 0 << 0; //TCP QTest::newRow("imap") << (int)QNetworkProxyQuery::TcpSocket << QUrl() << QString() << QString("qt-project.org") << 0 << (int)QNetworkProxy::TunnelingCapability; QTest::newRow("autobind-server") << (int)QNetworkProxyQuery::TcpServer << QUrl() << QString() << QString() << 0 << (int)QNetworkProxy::ListeningCapability; QTest::newRow("web-server") << (int)QNetworkProxyQuery::TcpServer << QUrl() << QString() << QString() << 80 << (int)QNetworkProxy::ListeningCapability; //windows: these should be bypassed if "bypass proxy server for local addresses" is ticked foreach (QHostAddress address, QNetworkInterface::allAddresses()) { QTest::newRow(qPrintable(address.toString())) << (int)QNetworkProxyQuery::TcpSocket << QUrl() << QString() << address.toString() << 0 << 0; } //UDP QTest::newRow("udp") << (int)QNetworkProxyQuery::UdpSocket << QUrl() << QString() << QString() << 0 << (int)QNetworkProxy::UdpTunnelingCapability; //Protocol tags QTest::newRow("http-tag") << (int)QNetworkProxyQuery::TcpSocket << QUrl() << QString("http") << QString("qt-project.org") << 80 << (int)QNetworkProxy::TunnelingCapability; QTest::newRow("ftp-tag") << (int)QNetworkProxyQuery::TcpSocket << QUrl() << QString("ftp") << QString("qt-project.org") << 21 << (int)QNetworkProxy::TunnelingCapability; QTest::newRow("https-tag") << (int)QNetworkProxyQuery::TcpSocket << QUrl() << QString("https") << QString("qt-project.org") << 443 << (int)QNetworkProxy::TunnelingCapability; #ifdef Q_OS_WIN //in Qt 4.8, "socks" would get the socks proxy, but we don't want to enforce that for all platforms QTest::newRow("socks-tag") << (int)QNetworkProxyQuery::TcpSocket << QUrl() << QString("socks") << QString("qt-project.org") << 21 << (int)(QNetworkProxy::TunnelingCapability | QNetworkProxy::ListeningCapability); #endif //windows: ssh is not a tag provided by the os, but any tunneling proxy is acceptable QTest::newRow("ssh-tag") << (int)QNetworkProxyQuery::TcpSocket << QUrl() << QString("ssh") << QString("qt-project.org") << 22 << (int)QNetworkProxy::TunnelingCapability; //Server protocol tags (ftp/http proxies are no good, we need socks or nothing) QTest::newRow("http-server-tag") << (int)QNetworkProxyQuery::TcpServer << QUrl() << QString("http") << QString() << 80 << (int)QNetworkProxy::ListeningCapability; QTest::newRow("ftp-server-tag") << (int)QNetworkProxyQuery::TcpServer << QUrl() << QString("ftp") << QString() << 21 << (int)QNetworkProxy::ListeningCapability; QTest::newRow("imap-server-tag") << (int)QNetworkProxyQuery::TcpServer << QUrl() << QString("imap") << QString() << 143 << (int)QNetworkProxy::ListeningCapability; //UDP protocol tag QTest::newRow("sip-udp-tag") << (int)QNetworkProxyQuery::UdpSocket << QUrl() << QString("sip") << QString("qt-project.org") << 5061 << (int)QNetworkProxy::UdpTunnelingCapability; } void tst_QNetworkProxyFactory::systemProxyForQuery() const { QFETCH(int, type); QFETCH(QUrl, url); QFETCH(QString, tag); QFETCH(QString, hostName); QFETCH(int, port); QFETCH(int, requiredCapabilities); QNetworkProxyQuery query; switch (type) { case QNetworkProxyQuery::UrlRequest: query = QNetworkProxyQuery(url); break; case QNetworkProxyQuery::TcpSocket: case QNetworkProxyQuery::UdpSocket: query = QNetworkProxyQuery(hostName, port, tag, QNetworkProxyQuery::QueryType(type)); break; case QNetworkProxyQuery::TcpServer: query = QNetworkProxyQuery(quint16(port), tag); break; } QElapsedTimer sw; sw.start(); QList systemProxyList = QNetworkProxyFactory::systemProxyForQuery(query); qDebug() << sw.elapsed() << "ms"; QVERIFY(!systemProxyList.isEmpty()); // for manual comparison with system qDebug() << systemProxyList; foreach (const QNetworkProxy &proxy, systemProxyList) { QVERIFY((requiredCapabilities == 0) || (proxy.capabilities() & requiredCapabilities)); } } void tst_QNetworkProxyFactory::systemProxyForQuery_local() { QList list; const QString proxyHost("myproxy.test.com"); // set an arbitrary proxy QNetworkProxy::setApplicationProxy(QNetworkProxy(QNetworkProxy::HttpProxy, proxyHost, 80)); factory = 0; // localhost list = QNetworkProxyFactory::proxyForQuery(QNetworkProxyQuery(QUrl("http://localhost/"))); QVERIFY(list.isEmpty() || (list[0].type() == QNetworkProxy::NoProxy)); list = QNetworkProxyFactory::proxyForQuery(QNetworkProxyQuery(QString("localhost"), 80)); QVERIFY(list.isEmpty() || (list[0].type() == QNetworkProxy::NoProxy)); // 127.0.0.1 list = QNetworkProxyFactory::proxyForQuery(QNetworkProxyQuery(QUrl("http://127.0.0.1/"))); QVERIFY(list.isEmpty() || (list[0].type() == QNetworkProxy::NoProxy)); list = QNetworkProxyFactory::proxyForQuery(QNetworkProxyQuery(QString("127.0.0.1"), 80)); QVERIFY(list.isEmpty() || (list[0].type() == QNetworkProxy::NoProxy)); // [::1] list = QNetworkProxyFactory::proxyForQuery(QNetworkProxyQuery(QUrl("http://[::1]/"))); QVERIFY(list.isEmpty() || (list[0].type() == QNetworkProxy::NoProxy)); list = QNetworkProxyFactory::proxyForQuery(QNetworkProxyQuery(QString("[::1]"), 80)); QVERIFY(list.isEmpty() || (list[0].type() == QNetworkProxy::NoProxy)); // an arbitrary host list = QNetworkProxyFactory::proxyForQuery(QNetworkProxyQuery(QUrl("http://another.host.com/"))); QVERIFY((!list.isEmpty()) && (list[0].hostName() == proxyHost)); list = QNetworkProxyFactory::proxyForQuery(QNetworkProxyQuery(QString("another.host.com"), 80)); QVERIFY((!list.isEmpty()) && (list[0].hostName() == proxyHost)); // disable proxy QNetworkProxy::setApplicationProxy(QNetworkProxy(QNetworkProxy::NoProxy)); factory = 0; // localhost list = QNetworkProxyFactory::proxyForQuery(QNetworkProxyQuery(QUrl("http://localhost/"))); QVERIFY(list.isEmpty() || (list[0].type() == QNetworkProxy::NoProxy)); list = QNetworkProxyFactory::proxyForQuery(QNetworkProxyQuery(QString("localhost"), 80)); QVERIFY(list.isEmpty() || (list[0].type() == QNetworkProxy::NoProxy)); // 127.0.0.1 list = QNetworkProxyFactory::proxyForQuery(QNetworkProxyQuery(QUrl("http://127.0.0.1/"))); QVERIFY(list.isEmpty() || (list[0].type() == QNetworkProxy::NoProxy)); list = QNetworkProxyFactory::proxyForQuery(QNetworkProxyQuery(QString("127.0.0.1"), 80)); QVERIFY(list.isEmpty() || (list[0].type() == QNetworkProxy::NoProxy)); // [::1] list = QNetworkProxyFactory::proxyForQuery(QNetworkProxyQuery(QUrl("http://[::1]/"))); QVERIFY(list.isEmpty() || (list[0].type() == QNetworkProxy::NoProxy)); list = QNetworkProxyFactory::proxyForQuery(QNetworkProxyQuery(QString("[::1]"), 80)); QVERIFY(list.isEmpty() || (list[0].type() == QNetworkProxy::NoProxy)); // an arbitrary host list = QNetworkProxyFactory::proxyForQuery(QNetworkProxyQuery(QUrl("http://another.host.com/"))); QVERIFY(list.isEmpty() || (list[0].type() == QNetworkProxy::NoProxy)); list = QNetworkProxyFactory::proxyForQuery(QNetworkProxyQuery(QString("another.host.com"), 80)); QVERIFY(list.isEmpty() || (list[0].type() == QNetworkProxy::NoProxy)); } #ifndef QT_NO_BEARERMANAGEMENT //Purpose of this test is just to check systemProxyForQuery doesn't hang or crash //with any given configuration including no configuration. //We can't test it returns the right proxies without implementing the native proxy code //again here, which would be testing our implementation against itself. //Therefore it's just testing that something valid is returned (at least a NoProxy entry) void tst_QNetworkProxyFactory::fromConfigurations() { QNetworkConfigurationManager manager; QList proxies; QUrl url(QLatin1String("http://qt.nokia.com")); //get from known configurations foreach (QNetworkConfiguration config, manager.allConfigurations()) { QNetworkProxyQuery query(config, url, QNetworkProxyQuery::UrlRequest); proxies = QNetworkProxyFactory::systemProxyForQuery(query); QVERIFY(!proxies.isEmpty()); foreach (QNetworkProxy proxy, proxies) { qDebug() << config.name() << " - " << config.identifier() << " - " << formatProxyName(proxy); } } //get from default configuration QNetworkProxyQuery defaultquery(url, QNetworkProxyQuery::UrlRequest); proxies = QNetworkProxyFactory::systemProxyForQuery(defaultquery); QVERIFY(!proxies.isEmpty()); foreach (QNetworkProxy proxy, proxies) { qDebug() << "default - " << formatProxyName(proxy); } //get from active configuration QNetworkSession session(manager.defaultConfiguration()); session.open(); QVERIFY(session.waitForOpened(30000)); proxies = QNetworkProxyFactory::systemProxyForQuery(defaultquery); QVERIFY(!proxies.isEmpty()); foreach (QNetworkProxy proxy, proxies) { qDebug() << "active - " << formatProxyName(proxy); } //get from known configurations while there is one active foreach (QNetworkConfiguration config, manager.allConfigurations()) { QNetworkProxyQuery query(config, url, QNetworkProxyQuery::UrlRequest); proxies = QNetworkProxyFactory::systemProxyForQuery(query); QVERIFY(!proxies.isEmpty()); foreach (QNetworkProxy proxy, proxies) { qDebug() << config.name() << " - " << config.identifier() << " - " << formatProxyName(proxy); } } } void tst_QNetworkProxyFactory::inNetworkAccessManager_data() { QTest::addColumn("config"); QTest::addColumn >("proxies"); QNetworkConfigurationManager manager; //get from known configurations foreach (QNetworkConfiguration config, manager.allConfigurations()) { QNetworkProxyQuery query(config, QUrl(QString("http://qt.nokia.com")), QNetworkProxyQuery::UrlRequest); QList proxies = QNetworkProxyFactory::systemProxyForQuery(query); QTest::newRow(config.name().toUtf8()) << config << proxies; } } //Purpose of this test is to check that QNetworkAccessManager uses the proxy from the configuration it //has been given. Needs two or more working configurations to be a good test. void tst_QNetworkProxyFactory::inNetworkAccessManager() { QFETCH(QNetworkConfiguration, config); QFETCH(QList, proxies); int count = QDebugProxyFactory::requestCounter; QNetworkAccessManager manager; manager.setConfiguration(config); //using an internet server, because cellular APs won't have a route to the test server. QNetworkRequest req(QUrl(QString("http://qt.nokia.com"))); QNetworkReply *reply = manager.get(req); connect(reply, SIGNAL(finished()), &QTestEventLoop::instance(), SLOT(exitLoop())); QTestEventLoop::instance().enterLoop(30); delete reply; if (count == QDebugProxyFactory::requestCounter) { //RND phones are preconfigured with several test access points which won't work without a matching SIM //If the network fails to start, QNAM won't ask the factory for proxies so we can't test. QSKIP("network configuration didn't start"); } QVERIFY(factory); qDebug() << "testing network configuration for" << config.name(); foreach (QNetworkProxy proxy, factory->returnedList) { qDebug() << formatProxyName(proxy); } qDebug() << " "; foreach (QNetworkProxy proxy, proxies) { qDebug() << formatProxyName(proxy); } if (config.type() != QNetworkConfiguration::InternetAccessPoint) QEXPECT_FAIL("","QNetworkProxyFactory::systemProxyForQuery doesn't work for service networks yet", Continue); QCOMPARE(factory->returnedList, proxies); } #endif //QT_NO_BEARERMANAGEMENT class QSPFQThread : public QThread { protected: virtual void run() { proxies = QNetworkProxyFactory::systemProxyForQuery(query); } public: QNetworkProxyQuery query; QList proxies; }; //regression test for QTBUG-18799 void tst_QNetworkProxyFactory::systemProxyForQueryCalledFromThread() { QUrl url(QLatin1String("http://qt.nokia.com")); QNetworkProxyQuery query(url); QSPFQThread thread; thread.query = query; connect(&thread, SIGNAL(finished()), &QTestEventLoop::instance(), SLOT(exitLoop())); thread.start(); QTestEventLoop::instance().enterLoop(5); QVERIFY(thread.isFinished()); QCOMPARE(thread.proxies, QNetworkProxyFactory::systemProxyForQuery(query)); } QTEST_MAIN(tst_QNetworkProxyFactory) #include "tst_qnetworkproxyfactory.moc"