qt5base-lts/tests/auto/network-settings.h
Thiago Macieira ac8fe3e645 QtNetworkSettings: narrow down hasIPv6 to specific OSes
It's currently conditional on two features used in QNetworkInterface.
The fact that QNetworkInterface misses those two features is probably
correlated to the failures observed on QNX, but is not the cause, so
this code was actually wrong and was possibly disabling the execution of
similar content on other OSes.

Therefore, this commit removes them and changes the conditional to
exclude the OS that is failing (QNX).

I find this situation unacceptable.  IPv6 support is mandatory for any
application after 2011-01-31, the date when IANA delegated its last IP
block, and definitely after 2019-11-25, when RIPE NCC ran completely
out. But since there's no SDK available for it, I'll grudgingly accept a
grandfathered exception because there's nothing I can do about it (I
tried to fix it; look at the change history of this patch set). I will
block any new OSes in that situation, though.

Task-number: QTBUG-116503
Change-Id: Ifa1111900d6945ea8e05fffd177ed6979c3e5916
Reviewed-by: Qt CI Bot <qt_ci_bot@qt-project.org>
Reviewed-by: Mårten Nordheim <marten.nordheim@qt.io>
2023-08-29 07:20:51 -07:00

319 lines
8.2 KiB
C++

// Copyright (C) 2019 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
#include <QString>
#include <QTest>
#include <QOperatingSystemVersion>
#include <QStringBuilder>
#ifdef QT_NETWORK_LIB
#include <QtNetwork/QHostInfo>
#include <QtNetwork/QHostAddress>
#include <QtNetwork/QAbstractSocket>
#include <QtNetwork/QTcpSocket>
#endif
#ifdef Q_OS_UNIX
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <unistd.h>
#endif
class QtNetworkSettings
{
public:
static QString serverLocalName()
{
return QString("qt-test-server");
}
static QString serverDomainName()
{
#ifdef QT_TEST_SERVER_DOMAIN
return QString(QT_TEST_SERVER_DOMAIN); // Defined in testserver feature
#else
return QString("qt-test-net");
#endif
}
static QString serverName()
{
return serverLocalName() + "." + serverDomainName();
}
static QString winServerName()
{
return serverName();
}
static QString wildcardServerName()
{
return "qt-test-server.wildcard.dev." + serverDomainName();
}
#ifdef QT_NETWORK_LIB
static QHostAddress getServerIpImpl(const QString &serverName)
{
const QHostInfo info = QHostInfo::fromName(serverName);
if (info.error()) {
QTest::qFail(qPrintable(info.errorString()), __FILE__, __LINE__);
return QHostAddress();
}
return info.addresses().constFirst();
}
static QHostAddress serverIP()
{
return getServerIpImpl(serverName());
}
#endif
static bool compareReplyIMAP(QByteArray const& actual)
{
// Server greeting may contain capability, version and server name
// But spec only requires "* OK" and "\r\n"
// Match against a prefix and postfix that covers all Cyrus versions
if (actual.startsWith("* OK ")
&& actual.endsWith("server ready\r\n")) {
return true;
}
return false;
}
static bool compareReplyIMAPSSL(QByteArray const& actual)
{
return compareReplyIMAP(actual);
}
static bool compareReplyFtp(QByteArray const& actual)
{
// output would be e.g. "220 (vsFTPd 2.3.5)\r\n221 Goodbye.\r\n"
QRegularExpression ftpVersion(QRegularExpression::anchoredPattern(QStringLiteral("220 \\(vsFTPd \\d+\\.\\d+.\\d+\\)\\r\\n221 Goodbye.\\r\\n")));
return ftpVersion.match(actual).hasMatch();
}
static bool hasIPv6()
{
#if defined(Q_OS_QNX)
// Qt's support for IPv6 on QNX appears to be broken.
// This is an unaccepable situation after 2011-01-31.
return false;
#elif defined(Q_OS_UNIX)
int s = ::socket(AF_INET6, SOCK_DGRAM, 0);
if (s == -1)
return false;
else {
struct sockaddr_in6 addr;
memset(&addr, 0, sizeof(addr));
addr.sin6_family = AF_INET6;
memcpy(&addr.sin6_addr, &in6addr_loopback, sizeof(in6_addr));
if (-1 == ::bind(s, (sockaddr*)&addr, sizeof(addr))) {
::close(s);
return false;
}
}
::close(s);
#endif
return true;
}
static bool canBindToLowPorts()
{
#ifdef Q_OS_UNIX
if (geteuid() == 0)
return true;
if (QOperatingSystemVersion::current() >= QOperatingSystemVersion::MacOSMojave)
return true;
// ### Which versions of iOS, watchOS and such does Apple's opening of
// all ports apply to?
return false;
#else
// Windows
return true;
#endif
}
#ifdef QT_NETWORK_LIB
static bool verifyTestNetworkSettings()
{
QHostInfo testServerResult = QHostInfo::fromName(QtNetworkSettings::serverName());
if (testServerResult.error() != QHostInfo::NoError) {
qWarning() << "Could not lookup" << QtNetworkSettings::serverName();
qWarning() << "Please configure the test environment!";
qWarning() << "See /etc/hosts or network-settings.h";
return false;
}
return true;
}
static bool verifyConnection(QString serverName, quint16 port, quint32 retry = 60)
{
QTcpSocket socket;
for (quint32 i = 1; i < retry; i++) {
socket.connectToHost(serverName, port);
if (socket.waitForConnected(1000))
return true;
// Wait for service to start up
QTest::qWait(1000);
}
socket.connectToHost(serverName, port);
return socket.waitForConnected(1000);
}
// Helper function for usage with QVERIFY2 on sockets.
static QByteArray msgSocketError(const QAbstractSocket &s)
{
QString result;
QDebug debug(&result);
debug.nospace();
debug.noquote();
if (!s.localAddress().isNull())
debug << "local=" << s.localAddress().toString() << ':' << s.localPort();
if (!s.peerAddress().isNull())
debug << ", peer=" << s.peerAddress().toString() << ':' << s.peerPort();
debug << ", type=" << s.socketType() << ", state=" << s.state()
<< ", error=" << s.error() << ": " << s.errorString();
return result.toLocal8Bit();
}
#endif // QT_NETWORK_LIB
static QString ftpServerName()
{
#ifdef QT_TEST_SERVER_NAME
return QString("vsftpd.") % serverDomainName();
#else
return serverName();
#endif
}
static QString ftpProxyServerName()
{
#ifdef QT_TEST_SERVER_NAME
return QString("ftp-proxy.") % serverDomainName();
#else
return serverName();
#endif
}
static QString httpServerName()
{
#ifdef QT_TEST_SERVER_NAME
return QString("apache2.") % serverDomainName();
#else
return serverName();
#endif
}
static QString httpProxyServerName()
{
#ifdef QT_TEST_SERVER_NAME
return QString("squid.") % serverDomainName();
#else
return serverName();
#endif
}
static QString socksProxyServerName()
{
#ifdef QT_TEST_SERVER_NAME
return QString("danted.") % serverDomainName();
#else
return serverName();
#endif
}
static QString imapServerName()
{
#ifdef QT_TEST_SERVER_NAME
return QString("cyrus.") % serverDomainName();
#else
return serverName();
#endif
}
static QString echoServerName()
{
#ifdef QT_TEST_SERVER_NAME
return QString("echo.") % serverDomainName();
#else
return serverName();
#endif
}
static QString firewallServerName()
{
#ifdef QT_TEST_SERVER_NAME
return QString("iptables.") % serverDomainName();
#else
return serverName();
#endif
}
static QString hostWithServiceOnPort(int port)
{
#if !defined(QT_TEST_SERVER)
Q_UNUSED(port);
return serverName();
#else
switch (port) {
case 13:
case 22:
case 139:
return serverName(); // No such things in docker (yet?)
case 7:
return echoServerName();
case 21:
return ftpServerName();
case 80:
case 443:
return httpServerName();
case 143:
return imapServerName();
case 3128:
case 3129:
case 3130:
return httpProxyServerName();
case 1080:
case 1081:
return socksProxyServerName();
case 2121:
return ftpProxyServerName();
default:
return serverName();
}
#endif // QT_TEST_SERVER
}
#ifdef QT_NETWORK_LIB
static QHostAddress imapServerIp()
{
return getServerIpImpl(imapServerName());
}
static QHostAddress httpServerIp()
{
return getServerIpImpl(httpServerName());
}
static QHostAddress httpProxyServerIp()
{
return getServerIpImpl(httpProxyServerName());
}
static QHostAddress socksProxyServerIp()
{
return getServerIpImpl(socksProxyServerName());
}
static QHostAddress ftpProxyServerIp()
{
return getServerIpImpl(ftpProxyServerName());
}
static QHostAddress ftpServerIp()
{
return getServerIpImpl(ftpServerName());
}
static QHostAddress firewallServerIp()
{
return getServerIpImpl(firewallServerName());
}
#endif // QT_NETWORK_LIB
};