diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index fbf101d28..4ab414b05 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -73,7 +73,7 @@ repos: hooks: - id: codespell name: Check spelling (codespell) - args: ["--ignore-words-list", "additionals,categor,curren,fo,ist,ket,notin,searchin,sectionin,superseeding,te,ths"] + args: ["--ignore-words-list", "additionals,categor,curren,fo,indexIn,ist,ket,notin,searchin,sectionin,superseeding,te,ths"] exclude: | (?x)^( .*\.desktop | diff --git a/src/base/http/requestparser.cpp b/src/base/http/requestparser.cpp index bd93fa6cc..f79b19e9e 100644 --- a/src/base/http/requestparser.cpp +++ b/src/base/http/requestparser.cpp @@ -279,7 +279,7 @@ bool RequestParser::parsePostMessage(const QByteArrayView data) // split data by "dash-boundary" const QByteArray dashDelimiter = QByteArray("--") + delimiter + CRLF; - QList multipart = splitToViews(data, dashDelimiter, Qt::SkipEmptyParts); + QList multipart = splitToViews(data, dashDelimiter); if (multipart.isEmpty()) { qWarning() << Q_FUNC_INFO << "multipart empty"; diff --git a/src/base/search/searchpluginmanager.cpp b/src/base/search/searchpluginmanager.cpp index 09b7a67cc..c43d35965 100644 --- a/src/base/search/searchpluginmanager.cpp +++ b/src/base/search/searchpluginmanager.cpp @@ -592,14 +592,14 @@ void SearchPluginManager::parseVersionInfo(const QByteArray &info) QHash updateInfo; int numCorrectData = 0; - const QList lines = Utils::ByteArray::splitToViews(info, "\n", Qt::SkipEmptyParts); + const QList lines = Utils::ByteArray::splitToViews(info, "\n"); for (QByteArrayView line : lines) { line = line.trimmed(); if (line.isEmpty()) continue; if (line.startsWith('#')) continue; - const QList list = Utils::ByteArray::splitToViews(line, ":", Qt::SkipEmptyParts); + const QList list = Utils::ByteArray::splitToViews(line, ":"); if (list.size() != 2) continue; const auto pluginName = QString::fromUtf8(list.first().trimmed()); diff --git a/src/base/utils/bytearray.cpp b/src/base/utils/bytearray.cpp index cae871f38..81e0a2627 100644 --- a/src/base/utils/bytearray.cpp +++ b/src/base/utils/bytearray.cpp @@ -30,28 +30,30 @@ #include "bytearray.h" #include +#include #include #include -QList Utils::ByteArray::splitToViews(const QByteArrayView in, const QByteArrayView sep, const Qt::SplitBehavior behavior) +QList Utils::ByteArray::splitToViews(const QByteArrayView in, const QByteArrayView sep) { + if (in.isEmpty()) + return {}; if (sep.isEmpty()) return {in}; + const QByteArrayMatcher matcher {sep}; QList ret; - ret.reserve((behavior == Qt::KeepEmptyParts) - ? (1 + (in.size() / sep.size())) - : (1 + (in.size() / (sep.size() + 1)))); - int head = 0; + ret.reserve(1 + (in.size() / (sep.size() + 1))); + qsizetype head = 0; while (head < in.size()) { - int end = in.indexOf(sep, head); + qsizetype end = matcher.indexIn(in, head); if (end < 0) end = in.size(); // omit empty parts - const QByteArrayView part = in.mid(head, (end - head)); - if (!part.isEmpty() || (behavior == Qt::KeepEmptyParts)) + const QByteArrayView part = in.sliced(head, (end - head)); + if (!part.isEmpty()) ret += part; head = end + sep.size(); diff --git a/src/base/utils/bytearray.h b/src/base/utils/bytearray.h index d0930dede..0f0d38fe7 100644 --- a/src/base/utils/bytearray.h +++ b/src/base/utils/bytearray.h @@ -37,8 +37,8 @@ class QByteArrayView; namespace Utils::ByteArray { - // Mimic QStringView(in).split(sep, behavior) - QList splitToViews(QByteArrayView in, QByteArrayView sep, Qt::SplitBehavior behavior = Qt::KeepEmptyParts); + // Inspired by QStringView(in).split(sep, Qt::SkipEmptyParts) + QList splitToViews(QByteArrayView in, QByteArrayView sep); QByteArray asQByteArray(QByteArrayView view); QByteArray toBase32(const QByteArray &in); diff --git a/src/base/utils/foreignapps.cpp b/src/base/utils/foreignapps.cpp index 3c80b12f0..f255a5607 100644 --- a/src/base/utils/foreignapps.cpp +++ b/src/base/utils/foreignapps.cpp @@ -68,7 +68,7 @@ namespace // Software 'Anaconda' installs its own python interpreter // and `python --version` returns a string like this: // "Python 3.4.3 :: Anaconda 2.3.0 (64-bit)" - const QList outputSplit = Utils::ByteArray::splitToViews(procOutput, " ", Qt::SkipEmptyParts); + const QList outputSplit = Utils::ByteArray::splitToViews(procOutput, " "); if (outputSplit.size() <= 1) return false; diff --git a/src/base/utils/password.cpp b/src/base/utils/password.cpp index b3be4dacd..fd3ffdfee 100644 --- a/src/base/utils/password.cpp +++ b/src/base/utils/password.cpp @@ -115,7 +115,7 @@ bool Utils::Password::PBKDF2::verify(const QByteArray &secret, const QString &pa bool Utils::Password::PBKDF2::verify(const QByteArray &secret, const QByteArray &password) { - const QList list = ByteArray::splitToViews(secret, ":", Qt::SkipEmptyParts); + const QList list = ByteArray::splitToViews(secret, ":"); if (list.size() != 2) return false; diff --git a/src/gui/search/searchwidget.cpp b/src/gui/search/searchwidget.cpp index 4369be0b1..4004d897c 100644 --- a/src/gui/search/searchwidget.cpp +++ b/src/gui/search/searchwidget.cpp @@ -172,8 +172,10 @@ namespace return nonstd::make_unexpected(readResult.error().message); } + const QList lines = Utils::ByteArray::splitToViews(readResult.value(), "\n"); QStringList history; - for (const QByteArrayView line : asConst(Utils::ByteArray::splitToViews(readResult.value(), "\n"))) + history.reserve(lines.size()); + for (const QByteArrayView line : lines) history.append(QString::fromUtf8(line)); return history; diff --git a/test/testutilsbytearray.cpp b/test/testutilsbytearray.cpp index 562c59020..39cf04a6c 100644 --- a/test/testutilsbytearray.cpp +++ b/test/testutilsbytearray.cpp @@ -1,5 +1,6 @@ /* * Bittorrent Client using Qt and libtorrent. + * Copyright (C) 2025 Mike Tzou (Chocobo1) * Copyright (C) 2023 Vladimir Golovnev * * This program is free software; you can redistribute it and/or @@ -26,10 +27,11 @@ * exception statement from your version. */ +#include +#include #include #include -#include "base/global.h" #include "base/utils/bytearray.h" class TestUtilsByteArray final : public QObject @@ -41,8 +43,50 @@ public: TestUtilsByteArray() = default; private slots: - void testBase32Encode() const + void testSplitToViews() const { + using BAViews = QList; + + const auto check = [](const QByteArrayView in, const QByteArrayView sep, const BAViews expected) + { + // verify it works + QCOMPARE(Utils::ByteArray::splitToViews(in, sep), expected); + + // verify it has the same behavior as `split(Qt::SkipEmptyParts)` + using Latin1Views = QList; + + const Latin1Views reference = QLatin1StringView(in) + .tokenize(QLatin1StringView(sep), Qt::SkipEmptyParts).toContainer(); + Latin1Views expectedStrings; + for (const auto &string : expected) + expectedStrings.append(QLatin1StringView(string)); + QCOMPARE(reference, expectedStrings); + }; + + check({}, {}, {}); + check({}, "/", {}); + check("/", "/", {}); + check("/a", "/", {"a"}); + check("/a/", "/", {"a"}); + check("/a/b", "/", (BAViews {"a", "b"})); + check("/a/b/", "/", (BAViews {"a", "b"})); + check("/a/b", "//", {"/a/b"}); + check("//a/b", "//", {"a/b"}); + check("//a//b", "//", (BAViews {"a", "b"})); + check("//a//b/", "//", (BAViews {"a", "b/"})); + check("//a//b//", "//", (BAViews {"a", "b"})); + check("///a//b//", "//", (BAViews {"/a", "b"})); + } + + void testAsQByteArray() const + { + QCOMPARE(Utils::ByteArray::asQByteArray(""), ""); + QCOMPARE(Utils::ByteArray::asQByteArray("12345"), "12345"); + } + + void testToBase32() const + { + QCOMPARE(Utils::ByteArray::toBase32({}), QByteArray()); QCOMPARE(Utils::ByteArray::toBase32(""), ""); QCOMPARE(Utils::ByteArray::toBase32("0123456789"), "GAYTEMZUGU3DOOBZ"); QCOMPARE(Utils::ByteArray::toBase32("ABCDE"), "IFBEGRCF");