5#include <QtNetwork/private/qtnetworkglobal_p.h>
7#if QT_CONFIG(topleveldomain)
9#include "QtCore/qfile.h"
10#include "QtCore/qloggingcategory.h"
11#include "QtCore/qstandardpaths.h"
12#include "QtCore/qstring.h"
14#if !QT_CONFIG(publicsuffix_qt) && !QT_CONFIG(publicsuffix_system)
15# error Enable at least one feature: publicsuffix-qt, publicsuffix-system
18#if QT_CONFIG(publicsuffix_qt)
19# include "psl_data.cpp"
23extern "C" int LookupStringInFixedSet(
const unsigned char *graph, std::size_t length,
24 const char *key, std::size_t key_length);
28using namespace Qt::StringLiterals;
30#if QT_CONFIG(publicsuffix_system)
31Q_STATIC_LOGGING_CATEGORY(lcTld,
"qt.network.tld")
34static constexpr int PSL_NOT_FOUND = -1;
35static constexpr int PSL_FLAG_EXCEPTION = 1 << 0;
36static constexpr int PSL_FLAG_WILDCARD = 1 << 1;
38class QPublicSuffixDatabase final
41#if QT_CONFIG(publicsuffix_system)
42 QPublicSuffixDatabase();
45 int lookupDomain(QByteArrayView domain)
const;
49#if QT_CONFIG(publicsuffix_qt)
51 kDafsa,
sizeof(kDafsa)
56#if QT_CONFIG(publicsuffix_system)
57 std::unique_ptr<QFile> m_dev;
59 bool loadFile(
const QString &fileName);
63int QPublicSuffixDatabase::lookupDomain(QByteArrayView domain)
const
65 return LookupStringInFixedSet(
reinterpret_cast<
const unsigned char *>(m_data.constData()),
66 m_data.size(), domain.data(), domain.size());
69#if QT_CONFIG(publicsuffix_system)
71static QStringList locatePublicSuffixFiles()
73 return QStandardPaths::locateAll(QStandardPaths::GenericDataLocation,
74 u"publicsuffix/public_suffix_list.dafsa"_s);
77QPublicSuffixDatabase::QPublicSuffixDatabase()
79 for (
auto &&fileName : locatePublicSuffixFiles()) {
80 if (loadFile(fileName))
84#if QT_CONFIG(publicsuffix_qt)
85 qCDebug(lcTld,
"Using builtin publicsuffix list");
87 qCWarning(lcTld,
"No usable publicsuffix file found");
91bool QPublicSuffixDatabase::loadFile(
const QString &fileName)
93 static const QByteArrayView DafsaFileHeader =
".DAFSA@PSL_0 \n";
95 qCDebug(lcTld,
"Loading publicsuffix file: %s", qUtf8Printable(fileName));
97 auto systemFile = std::make_unique<QFile>(fileName);
99 if (!systemFile->open(QIODevice::ReadOnly)) {
100 qCDebug(lcTld,
"Failed to open publicsuffix file: %s",
101 qUtf8Printable(systemFile->errorString()));
105 auto fileSize = systemFile->size();
107 if (fileSize < DafsaFileHeader.size() + 2) {
108 qCWarning(lcTld,
"publicsuffix file is too small: %zu", std::size_t(fileSize));
112 auto header = systemFile->read(DafsaFileHeader.size());
113 if (header != DafsaFileHeader) {
114 qCWarning(lcTld,
"Invalid publicsuffix file header: %s", header.toHex().constData());
119 if (!systemFile->seek(fileSize - 1)) {
120 qCWarning(lcTld,
"Failed to seek to the end of file: %s",
121 qUtf8Printable(systemFile->errorString()));
126 if (systemFile->read(&version, 1) != 1) {
127 qCWarning(lcTld,
"Failed to read publicsuffix version");
131 if (version != 0x01) {
132 qCWarning(lcTld,
"Unsupported publicsuffix version: %d",
int(version));
136 const auto dataSize = fileSize - DafsaFileHeader.size() - 1;
138 auto mappedData = systemFile->map(DafsaFileHeader.size(), dataSize);
140 qCDebug(lcTld,
"Using mapped system publicsuffix data");
142 m_data = QByteArrayView(mappedData, dataSize);
143 m_dev = std::move(systemFile);
147 qCDebug(lcTld,
"Failed to map publicsuffix file: %s",
148 qUtf8Printable(systemFile->errorString()));
150 systemFile->seek(DafsaFileHeader.size());
151 m_storage = systemFile->read(dataSize);
152 if (m_storage.size() != dataSize) {
153 qCWarning(lcTld,
"Failed to read publicsuffix file");
158 qCDebug(lcTld,
"Using system publicsuffix data");
164Q_GLOBAL_STATIC(QPublicSuffixDatabase, publicSuffix);
168static const QPublicSuffixDatabase m_publicSuffix;
173
174
175
176
177
178
180Q_NETWORK_EXPORT
bool qIsEffectiveTLD(QStringView domain)
187 QByteArray decodedDomain = domain.toUtf8();
188 QByteArrayView domainView(decodedDomain);
190#if QT_CONFIG(publicsuffix_system)
191 if (publicSuffix.isDestroyed())
194 auto publicSuffix = &m_publicSuffix;
197 auto ret = publicSuffix->lookupDomain(domainView);
198 if (ret != PSL_NOT_FOUND) {
199 if (ret & PSL_FLAG_EXCEPTION)
201 if ((ret & PSL_FLAG_WILDCARD) == 0)
205 const auto dot = domainView.indexOf(
'.');
207 return ret != PSL_NOT_FOUND;
208 ret = publicSuffix->lookupDomain(domainView.sliced(dot + 1));
209 if (ret == PSL_NOT_FOUND)
211 return (ret & PSL_FLAG_WILDCARD) != 0;